1 R Setup and Required Packages

In this project, the open-source R programming language is used to model the progression in the COVID-19 pandemic in different U.S. counties. R is maintained by an international team of developers who make the language available at The Comprehensive R Archive Network. Readers interested in reusing our code and reproducing our results should have R installed locally on their machines. R can be installed on a number of different operating systems (see Windows, Mac, and Linux for the installation instructions for these systems). We also recommend using the RStudio interface for R. The reader can download RStudio for free by following the instructions at the link. For non-R users, we recommend the Hands-on Programming with R for a brief overview of the software’s functionality. Hereafter, we assume that the reader has an introductory understanding of the R programming language.

In the code chunk below, we load the packages used to support our analysis. Note that the code of this and any of the code chunks can be hidden by clicking on the ‘Hide’ button to facilitate the navigation. The reader can hide all code and/or download the Rmd file associated with this document by clicking on the Code button on the top right corner of this document. Our input and output files can also be accessed/ downloaded from fmegahed/covid19-deaths.

# Check and install if these packages are not found locally on machine
if(require(pacman)==FALSE) install.packages("pacman") # check to see if the pacman package is installed; if not install it
if(require(devtools)==FALSE) install.packages("devtools") # check to see if the devtools package is installed; if not install it
if(require(urbnmapr)==FALSE) devtools::install_github('UrbanInstitute/urbnmapr')
if(require(albersusa)==FALSE) devtools::install_github('hrbrmstr/albersusa') #install package if needed

# Install (if needed) and load the following packages
pacman::p_load(tidyverse, magrittr, janitor, dataPreparation, lubridate, skimr, # for data analysis
               COVID19, rvest, # for extracting relevant data
               DT, pander, stargazer, knitr, # for formatting and nicely printed outputs
               scales, RColorBrewer, DataExplorer, tiff, grid,# for plots
               plotly, urbnmapr, albersusa, tigris, leaflet, tmap, # for maps
               zoo, fpp2, NbClust, # for TS analysis and clustering
               VIM, nnet, caret, MuMIn, # explanatory modeling
               conflicted) # for managing conflicts in functions with same names

# Handling conflicting function names from packages
conflict_prefer('combine', 'dplyr') # Preferring dplyr::combine over any other package
conflict_prefer('select', "dplyr") # Preferring dplyr::select over any other package
conflict_prefer("summarize", "dplyr") # Preferring dplyr::summarize over any other package
conflict_prefer("filter", "dplyr") # Preferring filter from dplyr
conflict_prefer("dist", "stats") # Preferring dist from stats
conflict_prefer("as.dist", "stats") # Preferring as.dist from stats

# Custom Functions
source_url('https://raw.githubusercontent.com/fmegahed/covid19-deaths/master/Markdown/custom_functions.R')

sInfo = sessionInfo() # saving all the packages/functions and session info

2 Extracting the Datasets

For our analysis, we fuse data from multiple sources. We describe the process of obtaining and merging each of these sources in the subsections below.

2.1 Time Series Data

In this section, we utilize the COVID19 package to obtain the following information: (Guidotti & Ardia, 2020)
- Confirmed cases, recoveries and deaths;
- Policy information (e.g., transport closing, school closing, closing event, movement restrictions, testing policies, and contact tracing); and
- Population and standard geographic information for each county.

From this information, we have also computed the new daily and weekly confirmed cases/deaths per county. The data is stored in a tidy format, but can be expanded to a wide format using pivot_wider() from the tidyverse package.

endDate = '2022-01-22'
endDatePrintV = format(ymd(endDate), format = "%b %d, %Y")

counties = covid19(
  country = "US", 
  level = 3, # for county
  start = "2020-03-01", # First Sunday in March
  end = endDate, # end Date 
  raw = FALSE, # to ensure that all counties have the same grid of dates
  amr = NULL, # we are not using the apple mobility data for our analysis
  gmr = NULL, # we are not using the Google mobility data for our analysis
  wb = NULL, # world bank data not helpful for county level analysis
  verbose = FALSE
  )

counties %<>% 
  # next line removes non-contiguous US states/territories
  filter(!administrative_area_level_2 %in% c('Alaska', 'Hawaii', 'Puerto Rico', 'Northern Mariana Islands', 'Virgin Islands')) %>% 
  # these are not counties
  filter(!is.na(key_local)) %>%  
  # grouping the data by the id column to make computations correct
  group_by(id) %>% 
  # to ensure correct calculations
  arrange(id, date) %>% 
  mutate(day = wday(date, label = TRUE) %>% factor(ordered = F), # day of week
         newCases = c(NA, diff(confirmed)), #  new daily cases per county
         newDeaths = c(NA, diff(deaths)) )  # new daily deaths per county

# manually identifying factor variables
factorVars = c("school_closing", "workplace_closing", "cancel_events",
               "gatherings_restrictions", "transport_closing", "stay_home_restrictions",
               "internal_movement_restrictions", "international_movement_restrictions",
               "information_campaigns", "testing_policy", "contact_tracing")

counties %<>% # converting those variables into character and then factor
  mutate_at(.vars = vars(any_of(factorVars)), .funs = as.character) %>% 
  mutate_at(.vars = vars(any_of(factorVars)), .funs = as.factor)


# saving the results as a RDS File
saveRDS(counties, '../Data/Output/counties.rds')

At this stage, we have only read the data based on the covid package. The resulting data is stored at an object titled counties, which contains 2,051,348 observations and 50 variables. Note that we have filtered observations that do not have a numeric key and removed some columns that do not add any value to future analysis (e.g., invariant cols).

2.2 Cross Sectional Data

In this analysis, we have extracted the following datasets:

  1. The CDC’s Social Vulnerability Index, where we would utilize the following four summary theme ranking variables:
    a. SocioeconomicRPL_THEME1
    b. Household Composition & DisabilityRPL_THEME2
    c. Minority Status & LanguageRPL_THEME3
    d. Housing Type & TransportationRPL_THEME4
    Note that each of these indices were computed by the Centers for Disease Control and Prevention/ Agency for Toxic Substances and Disease Registry/ Geospatial Research, Analysis, and Services Program. The csv file titled: SVI2018_US_COUNTY.csv was downloaded on Nov 02, 2020 (click here to download the file after selecting counties for the Geography Type and CSV File for the File type and clicking GO).

  2. We also extracted the following political and administrative variables, as of the beginning of the pandemic:
    a. Based on Data & Lab (2021), we have obtained the voting results for all counties in the 2016 Presidential elections. The data was used to compute the percentage of total votes that went to President Trump, with the underlying hypothesis that the politicization of COVID response (e.g., perception/willingness to use face masks, policies and the population’s reaction to the disease) may be explained by party affiliation.imputed its value with “Democratic” since D.C.’s Mayor is a Democrat (and DC is not a state).
    b. State’s CDC Region Classification: We have also engineered a region variable based on the CDC’s 10 Regions Framework. While geographic regions are hypothesized to be a factor in disease outbreaks, we chose to utilize the CDC regions specifically based on the following explanation from the aforementioned link: “CDC’s National Center for Chronic Disease Prevention and Health Promotion (NCCDPHP) is strengthening the consistency and quality of the guidance, communications, and technical assistance provided to states to improve coordination across our state programs”

  3. We also extracted an overall government response index capturing the strength of COVID-19 response policies on a state (and the District of Columbia) level from the Blavatnik School of Government’s GitHub Repository. This index captures 13 different indicators, capturing the ``full range of government response’’. Details for how this indicator is computed can be found at BSG-WP-2020/034.

# (1) Obtaining the svi data
svi <- read_csv("https://raw.githubusercontent.com/fmegahed/covid19-deaths/master/Data/Input/SVI2018_US_COUNTY.csv") %>% 
  clean_names() %>% # make everything lower_case
  select(location, fips, state, county, area_sqmi, e_totpop,
         rpl_theme1, rpl_theme2, rpl_theme3, rpl_theme4) %>% # columns of interest
  mutate(e_popdensity = e_totpop/ area_sqmi) %>% # computing population density
  filter(across(where(is.numeric), ~. >= 0)) %>%  # because NA's are coded as -999
  filter(!state %in% c('HAWAII', 'ALASKA')) # filtering to contiguous US


# (2) Political and Administrative Data
#### (a) County vote in the 2020 presidential elections
elections = read.csv("../Data/Input/countypres_2000-2020.csv") %>% # reading the downloaded CSV
  filter(year == 2020 & party == "REPUBLICAN") %>% # just keeping data for recent election and republican votes
  mutate(key_local = as.character(county_fips) %>% 
           str_pad(width = 5, side = 'left', pad = '0')) %>%
  group_by(key_local) %>% 
  summarise(
    # + absentee, advanced voting, prov, election day and absentee by mail votes
    candidatevotes = sum(candidatevotes),
    #  taking any of these values for total votes (it does not change by type)
    totalvotes = totalvotes,
    # calculating total votes received by the candidate
    percRepVotes = 100*(candidatevotes/totalvotes) ) %>% 
# computing percent of republican votes (from total votes)
  select(key_local, percRepVotes) %>% # keeping only the key and variable used in merge
  unique()


#### (b) State Region Classification
cdcRegions = data.frame(state = c('Connecticut', 'Maine', 'Massachusetts', 'New Hampshire', 'Rhode Island' , 
                                   'Vermont', 'New York', # End of Region A
                                   'Delaware', 'District of Columbia', 'Maryland', 'Pennsylvania',
                                   'Virginia', 'West Virginia', 'New Jersey', # End of Region B
                                   'North Carolina', 'South Carolina', 'Georgia', 'Florida', # Region C
                                   'Kentucky', 'Tennessee', 'Alabama', 'Mississippi', # Region D
                                   'Illinois', 'Indiana', 'Michigan', 'Minnesota', 'Ohio',
                                   'Wisconsin', # End of Region E
                                   'Arkansas', 'Louisiana', 'New Mexico', 'Oklahoma', 'Texas', # Region F
                                   'Iowa', 'Kansas', 'Missouri', 'Nebraska', # Region G
                                   'Colorado', 'Montana', 'North Dakota', 'South Dakota',
                                   'Utah', 'Wyoming', # End of Region H
                                   'Arizona', 'California', 'Hawaii', 'Nevada', # Region I
                                   'Alaska', 'Idaho', 'Oregon', 'Washington' # Region J
                                   ) %>% toupper(),
                        region = c(rep('A', 7), rep('B', 7), rep('C', 4),
                                    rep('D', 4), rep('E', 6), rep('F', 5),
                                    rep('G', 4), rep('H', 6), rep('I', 4),
                                    rep('J', 4) ) )

# (3) Policy Data
policy <- read_csv('https://raw.githubusercontent.com/OxCGRT/USA-covid-policy/master/data/OxCGRT_US_latest.csv')
policy <- filter(policy, !is.na(RegionName) | !RegionName %in% c('Alaska', 'Hawaii'))
policy$state <- toupper(policy$RegionName) # a state variable = an upper case of existing RegionName
policy$Date %<>% ymd() # converting the Date data to a date format

# Calculating a summary table of median value for the GovernmentResponseIndex per state
policySummary <- policy %>%
  filter(Date >= '2020-03-01' & Date <= endDate) %>% # to match our COVID Data timeSeries
  group_by(state) %>% # perform computations using the median value, per state, for each index
  summarise(GovernmentResponseIndexMedian = median(GovernmentResponseIndex, na.rm = TRUE))

# editing the name of Washington DC to District of Columbia to match the other datasets
policySummary$state %<>%  str_replace('WASHINGTON DC', 'DISTRICT OF COLUMBIA')


# (4) Combining all the potential predictors in a crossSectionDatal Frame
crossSectionalData <- left_join(svi, cdcRegions, by = 'state') %>%
  left_join(policySummary, by = 'state') %>% 
  rename(key_local = fips)

crossSectionalData <- left_join(crossSectionalData, elections,
                                by = 'key_local')

# saving the results as a RDS File
saveRDS(crossSectionalData, '../Data/Output/crossSectionalData.rds')

# Tabulating the results and providing a way to export the table to different formats
datatable(crossSectionalData %>% select(-c(state, county)),
          extensions = c('FixedColumns', 'Buttons'), options = list(
            dom = 'Bfrtip',
            scrollX = TRUE,
            buttons = c('copy', 'csv', 'excel', 'pdf'),
            fixedColumns = list(leftColumns = 2))
          ) %>% 
  formatRound(columns= c('area_sqmi', 'rpl_theme1', 'rpl_theme2',
                         'rpl_theme3', 'rpl_theme4', 'e_popdensity'),
              digits=2)

2.3 Exploratory Analysis

In this section, we perform an exploratory analysis on the data obtained from the multiple sources.

2.3.1 Cumulative Cases

noGoogleNAs <- filter(counties, !is.na(key_google_mobility)) # removing NAs from key_google_mobility
idIndex <- sample(noGoogleNAs$id, 9) # sampling 9 counties by id

# Saving cumulative deaths figure to an tiff file
tiff(filename = '../Figures/sampleCumulativeCases.tiff',
    width = 1366, height =768, pointsize = 16)
counties %>% filter(id %in% idIndex) %>% 
  ggplot(aes(x = date, y = confirmed, group = id, color = key_google_mobility)) +
  geom_line(size = 1.25) +
  scale_x_date(date_breaks = "1 month", date_labels = "%b") +
  facet_wrap(~ key_google_mobility, scales = 'free_y', ncol = 3) +
  theme(legend.position = 'none') + 
  labs(color = '', x = 'Month', y = 'Cumulative Cases By County') +
  scale_color_brewer(type = 'qual', palette = 'Paired')
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Creating an interactive plot for the markdown
p <- ggplot2::last_plot() + geom_line(size = 0.75) + # modifying the plot for plotly
  theme_bw(base_size = 9) + theme(legend.position = 'none') # to make margins smaller
ggplotly(p, height = 768) %>%  layout_ggplotly()

2.3.2 Cumulative Deaths

# Saving cumulative deaths figure to an tiff file
tiff(filename = '../Figures/sampleCumulativeDeaths.tiff',
    width = 1366, height =768, pointsize = 16)
counties %>% filter(id %in% idIndex) %>% 
  ggplot(aes(x = date, y = deaths, group = id, color = key_google_mobility)) +
  geom_line(size = 1.25) +
  scale_x_date(date_breaks = "1 month", date_labels = "%b") +
  facet_wrap(~ key_google_mobility, scales = 'free_y', ncol = 3) +
  theme(legend.position = 'none') + 
  labs(color = '', x = 'Month', y = 'Cumulative Deaths By County') +
  scale_color_brewer(type = 'qual', palette = 'Paired')
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Creating an interactive plot for the markdown
p <- ggplot2::last_plot() + geom_line(size = 0.75) + # modifying the plot for plotly
  theme_bw(base_size = 9) + theme(legend.position = 'none') # to make margins smaller
ggplotly(p, height = 768) %>%  layout_ggplotly()

2.3.3 RPL Theme 1

# Retrieving the U.S. county composite map as a simplefeature
cty_sf = get_urbn_map(map = "counties", sf = TRUE) %>% filter(!state_name %in% c('Alaska', 'Hawaii'))
cty_sf %<>% geo_join(crossSectionalData, by_sp= 'county_fips', by_df= 'key_local')

# Saving a static version of the figure as tiff
tiff(filename = '../Figures/rplTheme1Map.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(cty_sf) + tm_polygons('rpl_theme1', title = 'RPL Theme 1: Socioeconomic', palette = "-Greens")
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/rplTheme1Map.tiff") %>% grid.raster()

2.3.4 RPL Theme 2

# Saving a static version of the figure (capitalizing on the tmap package)
tiff(filename = '../Figures/rplTheme2Map.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(cty_sf) + tm_polygons('rpl_theme2', title = 'RPL Theme 2: Household Composition & Disability', palette = "-Greens")
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/rplTheme2Map.tiff") %>% grid.raster()

2.3.5 RPL Theme 3

# Saving a static version of the figure (capitalizing on the tmap package)
tiff(filename = '../Figures/rplTheme3Map.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(cty_sf) + tm_polygons('rpl_theme3', title = 'RPL Theme 3: Minority Status & Language', palette = "-Greens")
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/rplTheme3Map.tiff") %>% grid.raster()

2.3.6 RPL Theme 4

# Saving a static version of the figure (capitalizing on the tmap package)
tiff(filename = '../Figures/rplTheme4Map.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(cty_sf) + tm_polygons('rpl_theme4', title = 'RPL Theme 4: Housing Type & Transportation', palette = "-Greens")
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/rplTheme4Map.tiff") %>% grid.raster()

2.3.7 CDC Regions

crossSectionalData$state %<>% tolower() %>% tools::toTitleCase()
# Retrieving the U.S. state composite map as a simplefeature
state_sf = get_urbn_map(map = "states", sf = TRUE) %>% 
  filter(!state_name %in% c('Alaska', 'Hawaii') )
state_sf %<>% geo_join(crossSectionalData, by_sp= 'state_name', by_df= 'state')

# Saving a static version of the figure (capitalizing on the tmap package)
tiff(filename = '../Figures/cdcMap.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(state_sf) + tm_polygons('region', title = 'CDC Region', palette = "Paired")
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/cdcMap.tiff") %>% grid.raster()

2.3.8 Republican Votes in the 2020 Presidential Elections

# Saving a static version of the figure (capitalizing on the tmap package)
tiff(filename = '../Figures/governorsMap.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(state_sf) + 
  tm_polygons('percRepVotes', title = "2020 Republican Votes", palette = 'Reds')
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/governorsMap.tiff") %>% grid.raster()

2.3.9 Government Response Index Median Value

# Saving a static version of the figure (capitalizing on the tmap package)
tiff(filename = '../Figures/governmentResponseMap.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(state_sf) + tm_polygons('GovernmentResponseIndexMedian', title = "Median Value of the Government Response Index", palette = "-Greens")
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/governmentResponseMap.tiff") %>% grid.raster()

3 Time-Series Clustering

It is important to note that, in our estimation, there are three important decisions to be made when performing time-series clustering:

  • Preparation of the Different Time-Series to be Clustered In this section, we have (a) selected the new daily deaths per county as the primary variable of interest, (b) smoothed that variable using a seven-day moving average, and (c) scaled the observations within each county’s 7-day MA of new daily deaths such that it is bounded between 0 and 1. This allows us to compare the shape of the time-series/profile across counties of different populations and where the magnitude of the deaths is quite different.

  • Choice of Distance Measure: The Euclidean distance, The Euclidean Distance i.e., the \(l_2\) norm, is the most commonly used distance measure since it is computationally efficient. However, it may not be suitable for applications where the time-series are of different length in addition to being sensitive to noise, scale and time shifts (Sardá-Espinosa, 2017).

  • Choice of Clustering Algorithm: A large number of clustering algorithms have been proposed in the literature. Most common clustering approaches are shape-based, which include \(k-\)means clustering and hierarchical clustering. The reader is referred to Aghabozorgi et al. (2015) for a detailed review. In our preliminary analysis, we have chosen to use the hierarchical clustering approach since it provides an easy to understand dendogram and the number of counties was small. However, in our full analysis, we will use the \(k-\)means clustering algorithm since it is computationally efficient. Furthermore, we overcame the traditional limitation of having to pre-specify \(k\) by utilizing 26 indexes for determining the optimal number of clusters in a data set based on the excellent approach and package implementation of Charrad et al. (2014).

3.1 Data Preparation

clusteringPrep <- counties %>% # from the counties
  select(id, date, key_local, newDeaths) %>% # selecting minimal amount of cols for visual inspection
  arrange(id, date) %>% # arranged to ensure correct calculations
  mutate(newMA7 = rollmeanr(newDeaths, k = 7, fill = NA), # 7-day mm of new (adjusted) deaths
         maxMA7 = max(newMA7, na.rm = T), # obtaining the max per county to scale data
         scaledNewMA7 = pmax(0, newMA7/maxMA7, na.rm = TRUE) ) %>% # scaling data to a 0-1 scale by county
  select(id, key_local, date, scaledNewMA7) %>% # dropping the variable newDeaths
  pivot_wider(names_from = date, values_from = scaledNewMA7) # converting the data to a wide format for clustering

constantColumns  <- which_are_constant(clusteringPrep, verbose = F) # identifying constant columns
datesDropped <- colnames(clusteringPrep)[constantColumns] # used for printing the names after the code chunk

clusteringPrep %<>% select(-all_of(constantColumns) ) %>%  # speeds up clustering by dec length of series
  as.data.frame() # data needs to be data frame for clustering
row.names(clusteringPrep) = clusteringPrep[,1] # needed for tsclust

clusteringPrep = clusteringPrep[,-c(1)] %>% 
  select_if(~ !any(is.na(.))) # removing any date containing NA

# saving the results as a RDS File
saveRDS(clusteringPrep, '../Data/Output/clusteringPrep.rds')

The following dates were removed from our data frame since the scaledNEWMA7 variable was constant across all counties: .

3.2 Clustering Contiguous U.S. Counties

clusteringPrep %<>% select(-c(key_local))  # removing this variable so we can cluster

nc  <- NbClust(clusteringPrep, distance = "euclidean", # euclidean distance
             min.nc = 2, max.nc = 51, # searching for optimal k between k=2 and k=51
             method = "kmeans", # using the k-means method
             index = "all") # using 26 of the 30 indices in the package
## *** : The Hubert index is a graphical method of determining the number of clusters.
##                 In the plot of Hubert index, we seek a significant knee that corresponds to a 
##                 significant increase of the value of the measure i.e the significant peak in Hubert
##                 index second differences plot. 
## 
## *** : The D index is a graphical method of determining the number of clusters. 
##                 In the plot of D index, we seek a significant knee (the significant peak in Dindex
##                 second differences plot) that corresponds to a significant increase of the value of
##                 the measure. 
##  
## ******************************************************************* 
## * Among all indices:                                                
## * 8 proposed 2 as the best number of clusters 
## * 9 proposed 3 as the best number of clusters 
## * 3 proposed 28 as the best number of clusters 
## * 1 proposed 34 as the best number of clusters 
## * 1 proposed 43 as the best number of clusters 
## 
##                    ***** Conclusion *****                            
##  
## * According to the majority rule, the best number of clusters is  3 
##  
##  
## *******************************************************************
kclus  <- nc$Best.partition %>% as.data.frame() %>% #obtaining the best partition/ cluster assignment for optimal k
  rename(., cluster_group = .) %>% rownames_to_column("County") 

#converting the wide to tall data and adding the cluster groupings
clusters  <- clusteringPrep %>% 
  rownames_to_column(var = "County") %>% 
  pivot_longer(cols = starts_with("2020"), names_to = "Date") %>% 
  inner_join(., kclus, by = "County") %>% 
  mutate(cluster_group = as.factor(cluster_group))

idClusters  <- clusters %>% select(c(County, cluster_group)) # creating a look-up table of county and cluster group
colnames(idClusters)  <- c('id', 'cluster_group') # renaming the columns
idClusters %<>%  unique() #removing the duplicates due to different dates (we had that to ensure that the clustering was applied correctly)

# Adding Cluster Grouping to a subset of the counties data frame
clusterCounties <- counties %>% 
  select(c(id, key_local, key_google_mobility, administrative_area_level_2, administrative_area_level_3)) %>% 
  inner_join(., idClusters, by ='id') %>% 
  mutate(cluster_group = paste0('C', cluster_group)) %>% 
  unique()

# saving the results as a RDS File
saveRDS(clusterCounties, '../Data/Output/clusterCounties.rds')

3.3 Visualizing the Clustering Results

In this subsection, we provide three plots:

  • A paneled spaghetti plot, highlighting the median scaled time-series for profile for each cluster;
  • A panel plot where the first, second and third quartiles of the scaled time-series for each cluster are compared; and
  • An interactive chloropleth maps to visualize the spatial distribution of the clusters, where the reader can click on a given county to show: (a) county name, (b) assigned cluster, (c) population density, and (d) percentage of residents in poverty.

3.3.1 Spaghetti Plot

spaghettiDF <- counties %>% # from the counties
  select(id, date, newDeaths, key_google_mobility) %>% # selecting minimal columns
  left_join(clusterCounties[, c('id', 'cluster_group')], by = 'id') %>% # to get clusters
  arrange(id, date) %>% # arranged to ensure correct calculations
  mutate(newMA7 = rollmeanr(newDeaths, k = 7, fill = NA), # 7-day ma of new (adjusted) deaths
         maxMA7 = max(newMA7, na.rm = T), # obtaining the max per county to scale data
         scaledNewMA7 = pmax(0, newMA7/maxMA7, na.rm = TRUE) ) %>% 
  ungroup() %>% select(date, cluster_group, scaledNewMA7, key_google_mobility) %>% 
  group_by(date, cluster_group)

spaghettiDF$cluster_group %<>% as.factor() 

# Creating a Named Color Scale
colorPal <-  brewer.pal(n= levels(spaghettiDF$cluster_group) %>% length(), 'Set2')
names(colorPal) <- levels(spaghettiDF$cluster_group)

# Saving spaghetti plot to an tiff file
tiff(filename = '../Figures/spaghettiPlot.tiff', width = 1366, height =768, pointsize = 16)
spaghettiDF %>%  
  ggplot(aes(x = date, y = scaledNewMA7, color = cluster_group, group = key_google_mobility)) +
  scale_x_date(date_breaks = "1 month", date_labels = "%b") +
  geom_line(size = 0.25, alpha = 0.1) +
  stat_summary(aes(group = 1), 
               fun= median,
               geom = "line",
               size = 1.25, col = 'black') + 
  facet_wrap(~ cluster_group, ncol = 1) +
  theme(legend.position = 'none') + 
  labs(x = 'Month', y = 'Scaled New Deaths By Cluster By Day',
       caption = paste0('Solid black line represents the median for each cluster | 
       Based on Data from March 01, 2020 - ', endDatePrintV) )  +
  scale_color_manual(values = colorPal)
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/spaghettiPlot.tiff") %>% grid.raster()

3.3.2 Summary Plot

# Creating a data frame containing statistical summaries of the time series by cluster_group
summaryDf <- spaghettiDF %>% 
  summarise(Median = median(scaledNewMA7, na.rm= TRUE),
            `First Quartile` = quantile(scaledNewMA7, probs = 0.25, na.rm= TRUE),
            `Third Quartile` = quantile(scaledNewMA7, probs = 0.75, na.rm= TRUE)) %>% 
  pivot_longer(cols = c(`First Quartile`, Median, `Third Quartile`),
                        names_to = 'Statistic')

tiff(filename = '../Figures/summaryPlot.tiff', width = 1366, height =768, pointsize = 16)
summaryDf %>% 
  ggplot(aes(x = date, y = value, color = cluster_group, linetype =  Statistic)) +
  scale_x_date(date_breaks = "1 month", date_labels = "%b") +
  geom_line(size = 1.25) +
   scale_linetype_manual(values = c('dotted', 'solid', 'twodash')) +
  facet_wrap(~ cluster_group, ncol = 1) +
  theme(legend.position = 'top') + 
  labs(color = '', x = 'Month', y = 'Quartiles of Scaled New Deaths By Cluster By Day') +
  scale_color_manual(values = colorPal)
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/summaryPlot.tiff") %>% grid.raster()

3.3.3 Cluster Map

# Joining the clusterCounties results with the existing county simple features object (cty_sf)
clusterCounties %<>% ungroup()
cty_sf %<>% left_join(clusterCounties[, c('key_local', 'cluster_group')], by = c('county_fips'= 'key_local')) # adding cluster_group to cty_sf

# Creating a static visual for the paper
tiff(filename = '../Figures/clusterMap.tiff', width = 1366, height =768, pointsize = 16)
tm_shape(cty_sf) + tm_polygons('cluster_group', title = 'Cluster #', palette = colorPal)
invisible( dev.off() ) # to suppress the unwanted output from dev.off


# Creating an interactive visual Using the Leaflet Package
#### Creating a longlat projection (required by leaflet)
leaflet_sf <- counties_sf("longlat") %>% filter(!state %in% c('Alaska', 'Hawaii')) # from albersua
leaflet_sf %<>% geo_join(crossSectionalData, by_sp= 'fips', by_df= 'key_local') %>% 
  left_join(clusterCounties[, c('key_local', 'cluster_group')], by = c('fips' = 'key_local'))

#### Setting the Color Scheme
leafletPal <-  colorFactor('Set2', domain = leaflet_sf$cluster_group, na.color = "white")

#### The visual
leaflet(height=500) %>% # initializing the leaflet map
  setView(lng = -96, lat = 37.8, zoom = 3.8) %>% # setting the view on Continental US
  addTiles() %>% # adding the default tiles
  addPolygons(data = leaflet_sf, stroke = FALSE, fillColor = ~leafletPal(leaflet_sf$cluster_group), # adding the data
              weight = 2, opacity = 1, color = "white", dashArray = "3", fillOpacity = 0.7, # adding color specs
              popup = paste("County:", leaflet_sf$name, '<br>', 
                            "Cluster #:", leaflet_sf$cluster_group, '<br>',
                            "Population Density:", round(leaflet_sf$e_popdensity, 1), '<br>')) %>% #pop-up Menu
  addLegend(position = "bottomleft", pal = leafletPal, values =  leaflet_sf$cluster_group, 
            title = "Cluster #", opacity = 1) # legend formatting

4 Explanatory Modeling of Cluster Assignments

In the previous section, we showed that by using solely a scaled and smoothed time series of daily deaths per county, the counties are grouped into 3 categories (whose time-series have distinct shapes based on the Euclidean distance measure). In this section, we attempt to model the factors that are associated with the cluster assignment.

4.1 Descriptive Statistics

4.1.1 Skimming the Data

# Combining multiClass response with potential predictors
multiClassDF <- select(clusterCounties, key_local, cluster_group) %>% 
  left_join(crossSectionalData, by = 'key_local')  %>% 
  select(-c(e_totpop, area_sqmi, state, county))

saveRDS(multiClassDF, '../Data/Output/multiClassDF.rds') # saving the data

skim(multiClassDF) # printing a nice summary table of the data
Data summary
Name multiClassDF
Number of rows 3106
Number of columns 11
_______________________
Column type frequency:
character 4
numeric 7
________________________
Group variables None

Variable type: character

skim_variable n_missing complete_rate min max empty n_unique whitespace
key_local 0 1 5 5 0 3106 0
cluster_group 0 1 2 2 0 3 0
location 3 1 16 42 0 3103 0
region 3 1 1 1 0 10 0

Variable type: numeric

skim_variable n_missing complete_rate mean sd p0 p25 p50 p75 p100 hist
rpl_theme1 3 1 0.50 0.29 0.00 0.25 0.50 0.75 1.00 ▇▇▇▇▇
rpl_theme2 3 1 0.50 0.29 0.00 0.25 0.50 0.75 1.00 ▇▇▇▇▇
rpl_theme3 3 1 0.50 0.29 0.00 0.25 0.50 0.75 1.00 ▇▇▇▇▇
rpl_theme4 3 1 0.50 0.29 0.00 0.25 0.49 0.74 1.00 ▇▇▇▇▇
e_popdensity 3 1 240.81 1520.86 0.15 17.20 45.23 117.63 72052.99 ▇▁▁▁▁
GovernmentResponseIndexMedian 3 1 45.73 7.62 29.69 41.20 45.31 49.48 70.68 ▃▇▇▂▁
percRepVotes 6 1 65.12 15.98 8.73 55.88 68.39 77.52 96.18 ▁▂▅▇▃

4.1.2 Correlation Plot Among Continuous Predictors

tiff(filename = '../Figures/corrPlot.tiff',
    width = 1366, height =768, pointsize = 16)
na.omit(multiClassDF) %>% 
  plot_correlation(ggtheme = theme_bw(), type = 'c')  # compute corr among only continuous vars
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/corrPlot.tiff") %>% grid.raster()

4.1.3 Boxplot By Cluster

# Saving cumulative deaths figure to a tiff file
tiff(filename = '../Figures/boxPlot.tiff',
    width = 1366, height =768, pointsize = 16)
multiClassDF %>% 
  plot_boxplot(by = 'cluster_group', ncol = 2L, 
               ggtheme = theme_bw(),
               geom_boxplot_args = list('outlier.shape' = 1))
invisible( dev.off() ) # to suppress the unwanted output from dev.off

# Printing a png version of the plot in Markdown (lower quality image for quicker compilation of HTML)
readTIFF("../Figures/boxPlot.tiff") %>% grid.raster()

4.2 Explanatory Modeling Using Multinomial Regression

4.2.1 Model Building

df <- multiClassDF %>% select(-c(key_local, location)) %>% 
  mutate(cluster_group = as.factor(cluster_group))

df$e_popdensity <- log(df$e_popdensity) # since it is highly skewed and we are using a linear model 

# setting the reference level to category with max frequency
df$clustReLeveled <-  relevel(df$cluster_group, ref = maxCat(df$cluster_group) )
df  <-  df %>% select(-c(cluster_group)) # removed since we created a reLeveled version and stored it in a diff va

finalModel <-  quiet(multinom(clustReLeveled ~ ., data = df)) # create model

4.2.2 Resulting Model

# Saving the results as a latex table, but not printing it out in the Markdown document
invisible(stargazer(finalModel, type = 'latex', p.auto = FALSE, out="../Data/Output/multi.tex", 
                    single.row = TRUE, header = FALSE))
# tabulating model results as an HTML, which we print below
stargazer(finalModel, type = 'html', p.auto = FALSE, out="../Data/Output/multi.html", single.row = FALSE)
Dependent variable:
C1 C3
(1) (2)
rpl_theme1 -0.670** 0.829**
(0.304) (0.343)
rpl_theme2 0.360 0.492**
(0.235) (0.245)
rpl_theme3 0.573** 0.873***
(0.245) (0.264)
rpl_theme4 1.206*** 0.125
(0.236) (0.252)
e_popdensity 0.398*** 0.639***
(0.044) (0.050)
regionB -0.668** 0.970*
(0.284) (0.528)
regionC -3.544*** 2.036***
(0.483) (0.530)
regionD -1.336*** 1.667***
(0.338) (0.540)
regionE 0.190 -1.211*
(0.255) (0.648)
regionF -0.802** 2.287***
(0.317) (0.530)
regionG -1.580*** -1.348*
(0.324) (0.691)
regionH -0.694** 1.277**
(0.338) (0.588)
regionI -0.558 2.609***
(0.386) (0.565)
regionJ -0.787** 2.777***
(0.374) (0.552)
GovernmentResponseIndexMedian -0.003 0.005
(0.009) (0.009)
percRepVotes 0.010** 0.021***
(0.004) (0.005)
Constant -2.796*** -7.700***
(0.687) (0.885)
Akaike Inf. Crit. 4,866.208 4,866.208
Note: p<0.1; p<0.05; p<0.01

4.2.3 Model’s Performance

# examining how well the model performed on our entire dataset
# Recall that we are fitting an explanatory model, and not a predictive model
predictedClass <- predict(finalModel, df)

# Computing the Confusion Metrics and By Class Metrics
confMatrix = confusionMatrix(predictedClass, df$clustReLeveled)
saveRDS(confMatrix, '../Data/Output/confMatrix.rds') # saving the data

# Printing the Resulting Tables Nicely
pander(confMatrix$table)
  C2 C1 C3
C2 1263 344 276
C1 181 319 23
C3 206 91 397
pander(confMatrix$byClass)
Table continues below
  Sensitivity Specificity Pos Pred Value Neg Pred Value
Class: C2 0.7655 0.5724 0.6707 0.682
Class: C1 0.4231 0.913 0.6099 0.8312
Class: C3 0.5704 0.8765 0.572 0.8757
Table continues below
  Precision Recall F1 Prevalence Detection Rate
Class: C2 0.6707 0.7655 0.715 0.5323 0.4074
Class: C1 0.6099 0.4231 0.4996 0.2432 0.1029
Class: C3 0.572 0.5704 0.5712 0.2245 0.1281
  Detection Prevalence Balanced Accuracy
Class: C2 0.6074 0.6689
Class: C1 0.1687 0.6681
Class: C3 0.2239 0.7234
pander(confMatrix$overall)
Table continues below
Accuracy Kappa AccuracyLower AccuracyUpper AccuracyNull
0.6384 0.3823 0.6212 0.6553 0.5323
AccuracyPValue McnemarPValue
4.371e-33 8.025e-22

4.2.4 Visualizing the Model’s Predictions

predictedProbs <- fitted(finalModel) # computing predicted probabilities for each of the cluster outcome levels
mapResults <- cbind(na.omit(multiClassDF), predictedProbs) # col binding predProbs for Each Cluster with multiClassDF

# Finding indices to subset the data
numberOfClusters <- unique(mapResults$cluster_group) %>% as.character() %>% length() 
startCol <- ncol(mapResults) - numberOfClusters + 1
endCol <- ncol(mapResults)

# Finding whether the predicted and actual clusters matched for each county
mapResults$LargestProbCluster <- colnames(mapResults[, startCol:endCol])[apply(mapResults[, startCol:endCol], 1, which.max)] 
mapResults$match <- ifelse(mapResults$cluster_group == mapResults$LargestProbCluster, 'Yes', 'No') %>% as.factor()

# Retrieving the U.S. county composite map as a simplefeature (since it has been overwritten)
cty_sf = get_urbn_map(map = "counties", sf = TRUE) %>% 
  filter(!state_name %in% c('Alaska', 'Hawaii'))

cty_sf %<>% geo_join(mapResults, by_sp= 'county_fips', by_df= 'key_local')

# Creating a static visual for the paper
pdf(file = '../Figures/clusterMatchMap.pdf', width = 6.5, height = 6.5)
tm_shape(cty_sf) + 
  tm_polygons('match', title = 'Cluster Match', style = 'cont', palette = "div") +
  tm_layout(aes.palette = list(div = list("Yes" = "#FFFFFF", "Missing" = "#BDBDBD", "No" = "#000000"))) +
  tm_layout(outer.margins = c(0,0,0,0), frame = F, legend.text.size = 1)
invisible( dev.off() ) # to suppress the unwanted output from dev.off


# Creating an interactive visual Using the Leaflet Package
#### Creating a longlat projection (required by leaflet)
leaflet_sf <- counties_sf("longlat") %>% filter(!state %in% c('Alaska', 'Hawaii')) # from albersua
leaflet_sf %<>% geo_join(mapResults, by_sp= 'fips', by_df= 'key_local')

#### Setting the Color Scheme
leafletPal <-  colorFactor(palette = c("#CAB2D6", "#6A3D9A"), levels = c('Yes', 'No'), na.color = "white")

#### The visual
leaflet(height=500) %>% # initializing the leaflet map
  setView(lng = -96, lat = 37.8, zoom = 3.8) %>% # setting the view on Continental US
  addTiles() %>% # adding the default tiles
  addPolygons(data = leaflet_sf, stroke = FALSE, fillColor = ~leafletPal(leaflet_sf$match), # adding the data
              weight = 2, opacity = 1, color = "white", dashArray = "3", fillOpacity = 0.7, # adding color specs
              popup = paste("County:", leaflet_sf$name, '<br>', 
                            "Cluster #:", leaflet_sf$cluster_group, '<br>',
                            "Cluster Predicted:", leaflet_sf$LargestProbCluster, '<br>',
                            "Cluster Match:", leaflet_sf$match, '<br>')) %>% #pop-up Menu
  addLegend(position = "bottomleft", pal = leafletPal, values =  leaflet_sf$match, 
            title = "Cluster Match", opacity = 1) # legend formatting

4.2.5 A Note on Our Predictive Performance

In the code chunk below, we capitalize on the MuMIn package to generate a model selection table of models with combinations (subsets) of fixed effect terms in the global model. More specifically, the code below identifies the combination of predictors (and hence the number of model parameters) that would minimize the AIC.

dfNoNAs <- na.omit(df) # since the MuMIn package expects no NAs in the data frame
fm <-  quiet(multinom(clustReLeveled ~ ., data = dfNoNAs, na.action = na.fail)) # create model
ms <- quiet(dredge(fm, rank = 'AIC')) # generating the model selection table

# Printing the models with the lowest AIC first
datatable(ms %>% as.data.frame() %>% select(-c(delta, weight)),
          extensions = c('FixedColumns', 'Buttons'), options = list(
            dom = 'Bfrtip',
            scrollX = TRUE,
            buttons = c('copy', 'csv', 'excel', 'pdf', 'print'),
            fixedColumns = list(leftColumns = 0, rightColumns = 3)),
          rownames = FALSE) %>% 
  formatRound(columns= c('logLik', 'AIC'), digits=0)

With the use of an alternative model selection criterion and this evaluation, we confirm that the optimal multinomial regression model:

  • includes the variables denoted with a ‘+’ sign (in the first row of the printed table);
  • contains 32 parameters in the model;
  • results in a logLik values of -2399; and
  • results in a minimum AIC of 4863.

5 Concluding Remarks

In this R Markdown document, we have shown that our proposed two stage framework for modeling the smoothed and scaled time series of new daily cases can provide insights into the shape of the outbreak’s time-series and some of its associated factors. Specifically, we have shown that:

  • On a county-level, the time series of COVID-19 new daily cases can be clustered into 3 clusters.

  • Using a multinomial regression model, we have shown/quantified the impact of the following factors: rpl_theme1, rpl_theme2, rpl_theme3, rpl_theme4, e_popdensity, region, GovernmentResponseIndexMedian and percRepVotes on the odds of being in a specific cluster when compared to the baseline.


6 References

Aghabozorgi, S., Shirkhorshidi, A. S., & Wah, T. Y. (2015). Time-series clustering–a decade review. Information Systems, 53, 16–38.
Charrad, M., Ghazzali, N., Boiteau, V., & Niknafs, A. (2014). NbClust: An r package for determining the relevant number of clusters in a data set. Journal of Statistical Software, Articles, 61(6), 1–36. https://doi.org/10.18637/jss.v061.i06
Data, M. E., & Lab, S. (2021). County Presidential Election Returns 2000-2020 (Version V9) [Data set]. Harvard Dataverse. https://doi.org/10.7910/DVN/VOQCHQ
Guidotti, E., & Ardia, D. (2020). COVID-19 data hub. Journal of Open Source Software, 5(51), 2376. https://doi.org/10.21105/joss.02376

7 Appendix

In this appendix, we print all the R packages used in our analysis and their versions to assist with reproducing our results/analysis.

pander(sessionInfo(), compact = TRUE) # printing the session info

R version 4.1.2 (2021-11-01)

Platform: x86_64-w64-mingw32/x64 (64-bit)

locale: LC_COLLATE=English_United States.1252, LC_CTYPE=English_United States.1252, LC_MONETARY=English_United States.1252, LC_NUMERIC=C and LC_TIME=English_United States.1252

attached base packages: grid, stats, graphics, grDevices, utils, datasets, methods and base

other attached packages: conflicted(v.1.1.0), MuMIn(v.1.43.17), caret(v.6.0-90), lattice(v.0.20-45), nnet(v.7.3-17), VIM(v.6.1.1), colorspace(v.2.0-2), NbClust(v.3.0), expsmooth(v.2.3), fma(v.2.4), forecast(v.8.16), fpp2(v.2.4), zoo(v.1.8-9), tmap(v.3.3-2), leaflet(v.2.0.4.1), tigris(v.1.5), plotly(v.4.10.0), tiff(v.0.1-10), DataExplorer(v.0.8.2), scales(v.1.1.1), knitr(v.1.37), stargazer(v.5.2.2), pander(v.0.6.4), DT(v.0.20), rvest(v.1.0.2), COVID19(v.3.0.1), skimr(v.2.1.3), dataPreparation(v.1.0.4), progress(v.1.2.2), Matrix(v.1.4-0), lubridate(v.1.8.0), janitor(v.2.1.0), magrittr(v.2.0.1), forcats(v.0.5.1), stringr(v.1.4.0), dplyr(v.1.0.7), purrr(v.0.3.4), readr(v.2.1.1), tidyr(v.1.1.4), tibble(v.3.1.6), tidyverse(v.1.3.1), albersusa(v.0.4.1), urbnmapr(v.0.0.0.9002), devtools(v.2.4.3), usethis(v.2.1.5), pacman(v.0.5.1), RColorBrewer(v.1.1-2) and ggplot2(v.3.3.5)

loaded via a namespace (and not attached): utf8(v.1.2.2), R.utils(v.2.11.0), tidyselect(v.1.1.1), htmlwidgets(v.1.5.4), ranger(v.0.13.1), maptools(v.1.1-2), pROC(v.1.18.0), munsell(v.0.5.0), codetools(v.0.2-18), units(v.0.7-2), future(v.1.23.0), withr(v.2.4.3), highr(v.0.9), uuid(v.1.0-3), rstudioapi(v.0.13), stats4(v.4.1.2), robustbase(v.0.93-9), vcd(v.1.4-9), listenv(v.0.8.0), TTR(v.0.24.3), labeling(v.0.4.2), repr(v.1.1.4), farver(v.2.1.0), bit64(v.4.0.5), rprojroot(v.2.0.2), parallelly(v.1.30.0), vctrs(v.0.3.8), generics(v.0.1.1), ipred(v.0.9-12), xfun(v.0.29), R6(v.2.5.1), cachem(v.1.0.6), assertthat(v.0.2.1), vroom(v.1.5.7), networkD3(v.0.4), rgeos(v.0.5-9), gtable(v.0.3.0), globals(v.0.14.0), lwgeom(v.0.2-8), processx(v.3.5.2), timeDate(v.3043.102), rlang(v.0.4.12), splines(v.4.1.2), rgdal(v.1.5-28), lazyeval(v.0.2.2), ModelMetrics(v.1.2.2.2), dichromat(v.2.0-0), broom(v.0.7.11), reshape2(v.1.4.4), yaml(v.2.2.1), abind(v.1.4-5), modelr(v.0.1.8), crosstalk(v.1.2.0), backports(v.1.4.1), quantmod(v.0.4.18), lava(v.1.6.10), tools(v.4.1.2), ellipsis(v.0.3.2), raster(v.3.5-11), jquerylib(v.0.1.4), proxy(v.0.4-26), sessioninfo(v.1.2.2), plyr(v.1.8.6), Rcpp(v.1.0.8), base64enc(v.0.1-3), classInt(v.0.4-3), ps(v.1.6.0), prettyunits(v.1.1.1), rpart(v.4.1-15), fracdiff(v.1.5-1), tmaptools(v.3.1-1), haven(v.2.4.3), fs(v.1.5.2), leafem(v.0.1.6), data.table(v.1.14.2), lmtest(v.0.9-39), reprex(v.2.0.1), pkgload(v.1.2.4), hms(v.1.1.1), evaluate(v.0.14), XML(v.3.99-0.8), readxl(v.1.3.1), gridExtra(v.2.3), testthat(v.3.1.2), compiler(v.4.1.2), KernSmooth(v.2.23-20), crayon(v.1.4.2), R.oo(v.1.24.0), htmltools(v.0.5.2), tzdb(v.0.2.0), DBI(v.1.1.2), dbplyr(v.2.1.1), MASS(v.7.3-55), rappdirs(v.0.3.3), sf(v.1.0-5), boot(v.1.3-28), car(v.3.0-12), brio(v.1.1.3), cli(v.3.1.1), R.methodsS3(v.1.8.1), quadprog(v.1.5-8), gower(v.0.2.2), parallel(v.4.1.2), igraph(v.1.2.11), pkgconfig(v.2.0.3), foreign(v.0.8-82), laeken(v.0.5.2), sp(v.1.4-6), recipes(v.0.1.17), terra(v.1.5-12), xml2(v.1.3.3), foreach(v.1.5.1), bslib(v.0.3.1), prodlim(v.2019.11.13), snakecase(v.0.11.0), callr(v.3.7.0), digest(v.0.6.29), rmarkdown(v.2.11), cellranger(v.1.1.0), leafsync(v.0.1.0), curl(v.4.3.2), urca(v.1.3-0), lifecycle(v.1.0.1), nlme(v.3.1-155), jsonlite(v.1.7.3), tseries(v.0.10-49), carData(v.3.0-5), desc(v.1.4.0), viridisLite(v.0.4.0), fansi(v.1.0.2), pillar(v.1.6.4), survival(v.3.2-13), fastmap(v.1.1.0), httr(v.1.4.2), DEoptimR(v.1.0-10), pkgbuild(v.1.3.1), glue(v.1.6.0), xts(v.0.12.1), remotes(v.2.4.2), iterators(v.1.0.13), png(v.0.1-7), leaflet.providers(v.1.9.0), bit(v.4.0.4), class(v.7.3-20), stringi(v.1.7.6), sass(v.0.4.0), stars(v.0.5-5), memoise(v.2.0.1), future.apply(v.1.8.1) and e1071(v.1.7-9)


  1. Email: | Phone: +1-513-529-4185 | Website: Miami University Official↩︎

  2. Email: | Phone: +1-513-529-4823 | Website: Miami University Official↩︎

  3. Email: | Website: Saint Louis University Official↩︎

LS0tDQp0aXRsZTogIkEgVHdvLVN0YWdlIE1vZGVsaW5nIEZyYW1ld29yayBmb3IgQW5hbHl6aW5nIENPVklELTE5IERlYXRocyBCeSBDb3VudHkiDQphdXRob3I6DQogIC0gbmFtZTogIkZhZGVsIE0uIE1lZ2FoZWQgXltFbWFpbDogZm1lZ2FoZWRAbWlhbWlvaC5lZHUgfCBQaG9uZTogKzEtNTEzLTUyOS00MTg1IHwgV2Vic2l0ZTogPGEgaHJlZj1cImh0dHBzOi8vbWlhbWlvaC5lZHUvZnNiL2RpcmVjdG9yeS8/dXA9L2RpcmVjdG9yeS9tZWdhaGVmbVwiPk1pYW1pIFVuaXZlcnNpdHkgT2ZmaWNpYWw8L2E+XSINCiAgICBhZmZpbGlhdGlvbjogRmFybWVyIFNjaG9vbCBvZiBCdXNpbmVzcywgTWlhbWkgVW5pdmVyc2l0eQ0KICAtIG5hbWU6ICJBbGxpc29uIEpvbmVzLUZhcm1lciBeW0VtYWlsOiBmYXJtZXJsMkBtaWFtaW9oLmVkdSB8IFBob25lOiArMS01MTMtNTI5LTQ4MjMgfCBXZWJzaXRlOiA8YSBocmVmPVwiaHR0cHM6Ly9taWFtaW9oLmVkdS9mc2IvZGlyZWN0b3J5Lz91cD0vZGlyZWN0b3J5L2Zhcm1lcmwyXCI+TWlhbWkgVW5pdmVyc2l0eSBPZmZpY2lhbDwvYT5dIg0KICAgIGFmZmlsaWF0aW9uOiBGYXJtZXIgU2Nob29sIG9mIEJ1c2luZXNzLCBNaWFtaSBVbml2ZXJzaXR5DQogIC0gbmFtZTogIlN0ZXZlIFJpZ2RvbiBeW0VtYWlsOiBzdGV2ZS5yaWdkb25Ac2x1LmVkdSB8IFdlYnNpdGU6IDxhIGhyZWY9XCJodHRwczovL3d3dy5zbHUuZWR1L3B1YmxpYy1oZWFsdGgtc29jaWFsLWp1c3RpY2UvZmFjdWx0eS9yaWdkb24tc3RldmVuLnBocFwiPlNhaW50IExvdWlzIFVuaXZlcnNpdHkgT2ZmaWNpYWw8L2E+XSINCiAgICBhZmZpbGlhdGlvbjogQ29sbGVnZSBvZiAgUHVibGljIEhlYWx0aCBhbmQgU29jaWFsIEp1c3RpY2UsIFNhaW50IExvdWlzIFVuaXZlcnNpdHkNCmJpYmxpb2dyYXBoeTogY292aWRSZWZzLmJpYg0KY3NsOiBhcGEuY3NsDQpkYXRlOiAiYHIgZm9ybWF0KFN5cy50aW1lKCksICclQiAlZCwgJVknKWAiDQpvdXRwdXQ6IA0KICBodG1sX2RvY3VtZW50Og0KICAgIHRvYzogVFJVRQ0KICAgIHRvY19mbG9hdDogVFJVRQ0KICAgIG51bWJlcl9zZWN0aW9uczogVFJVRQ0KICAgIHRoZW1lOiBzaW1wbGV4DQogICAgcGFnZWRfZGY6IFRSVUUNCiAgICBjb2RlX2ZvbGRpbmc6IHNob3cNCiAgICBjb2RlX2Rvd25sb2FkOiBUUlVFDQogIGluY2x1ZGVzOg0KICAgIGluX2hlYWRlcjogc3RydWN0dXJlLnRleA0KLS0tDQoNCmBgYHtyIHNldHVwLCBpbmNsdWRlPUZBTFNFfQ0Ka25pdHI6Om9wdHNfY2h1bmskc2V0KGNhY2hlID0gVFJVRSwNCiAgICAgICAgICAgICAgICAgICAgICBlY2hvID0gVFJVRSwNCiAgICAgICAgICAgICAgICAgICAgICB3YXJuaW5nID0gRkFMU0UsDQogICAgICAgICAgICAgICAgICAgICAgbWVzc2FnZSA9IEZBTFNFLA0KICAgICAgICAgICAgICAgICAgICAgIHByb2dyZXNzID0gRkFMU0UsIA0KICAgICAgICAgICAgICAgICAgICAgIHZlcmJvc2UgPSBGQUxTRSwNCiAgICAgICAgICAgICAgICAgICAgICBkZXYgPSBjKCdwbmcnLCAncGRmJywgJ3RpZmYnLCAncG9zdHNjcmlwdCcpLA0KICAgICAgICAgICAgICAgICAgICAgIGZpZy5yZXRpbmEgPSAyLA0KICAgICAgICAgICAgICAgICAgICAgIG91dC53aWR0aCA9ICcxMDAlJywNCiAgICAgICAgICAgICAgICAgICAgICBmaWcuYXNwID0gMC43KQ0KDQojIFNldHRpbmcgcHJvcGVydGllcyBmb3IgdGhlIGRlZmF1bHQgdGhlbWVfYncoKSBiZWhhdmlvciBmb3IgYWxsIHBsb3RzDQppZihyZXF1aXJlKGdncGxvdDIpID09IEZBTFNFKSBpbnN0YWxsLnBhY2thZ2VzKCJnZ3Bsb3QyIikNCmxpYnJhcnkoZ2dwbG90MikgOyB0aGVtZV9zZXQodGhlbWVfYncoYmFzZV9zaXplID0gMTEpICsgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gJ3RvcCcpKSANCg0KIyBTZXR0aW5nIGRlZmF1bHQgY29sb3IgcGFsZXR0ZXMgdG8gUkNvbG9yQnJld2VyIFBhbGV0dGVzDQppZihyZXF1aXJlKFJDb2xvckJyZXdlcikgPT0gRkFMU0UpIGluc3RhbGwucGFja2FnZXMoIlJDb2xvckJyZXdlciIpDQpzY2FsZV9jb2xvdXJfZGlzY3JldGUgPSBzY2FsZV9jb2xvcl9icmV3ZXIocGFsZXR0ZSA9ICJEYXJrMiIpDQoNCiMgU2V0dGluZyB0aGUgcmFuZG9tIHNlZWQgYW5kIGNodW5rIGRlcGVuZGVuY2llcw0Ka25pdHI6Om9wdHNfY2h1bmskc2V0KGNhY2hlLmV4dHJhID0gc2V0LnNlZWQoMjAyMiksDQogICAgICAgICAgICAgICAgICAgICAgYXV0b2RlcCA9IFRSVUUpIA0Ka25pdHI6OmRlcF9hdXRvKCkNCmBgYA0KDQoNCiMgUiBTZXR1cCBhbmQgUmVxdWlyZWQgUGFja2FnZXMNCkluIHRoaXMgcHJvamVjdCwgdGhlIG9wZW4tc291cmNlIFIgcHJvZ3JhbW1pbmcgbGFuZ3VhZ2UgaXMgdXNlZCB0byBtb2RlbCB0aGUgcHJvZ3Jlc3Npb24gaW4gdGhlIENPVklELTE5IHBhbmRlbWljIGluIGRpZmZlcmVudCBVLlMuIGNvdW50aWVzLiBSIGlzIG1haW50YWluZWQgYnkgYW4gaW50ZXJuYXRpb25hbCB0ZWFtIG9mIGRldmVsb3BlcnMgd2hvIG1ha2UgdGhlIGxhbmd1YWdlIGF2YWlsYWJsZSBhdCBbVGhlIENvbXByZWhlbnNpdmUgUiBBcmNoaXZlIE5ldHdvcmtdKGh0dHBzOi8vY3Jhbi5yLXByb2plY3Qub3JnLykuIFJlYWRlcnMgaW50ZXJlc3RlZCBpbiByZXVzaW5nIG91ciBjb2RlIGFuZCByZXByb2R1Y2luZyBvdXIgcmVzdWx0cyBzaG91bGQgaGF2ZSBSIGluc3RhbGxlZCBsb2NhbGx5IG9uIHRoZWlyIG1hY2hpbmVzLiBSIGNhbiBiZSBpbnN0YWxsZWQgb24gYSBudW1iZXIgb2YgZGlmZmVyZW50IG9wZXJhdGluZyBzeXN0ZW1zIChzZWUgW1dpbmRvd3NdKGh0dHBzOi8vY3Jhbi5yLXByb2plY3Qub3JnL2Jpbi93aW5kb3dzLyksIFtNYWNdKGh0dHBzOi8vY3Jhbi5yLXByb2plY3Qub3JnL2Jpbi9tYWNvc3gvKSwgYW5kIFtMaW51eF0oaHR0cHM6Ly9jcmFuLnItcHJvamVjdC5vcmcvYmluL2xpbnV4LykgZm9yIHRoZSBpbnN0YWxsYXRpb24gaW5zdHJ1Y3Rpb25zIGZvciB0aGVzZSBzeXN0ZW1zKS4gV2UgYWxzbyByZWNvbW1lbmQgdXNpbmcgdGhlIFJTdHVkaW8gaW50ZXJmYWNlIGZvciBSLiBUaGUgcmVhZGVyIGNhbiBbZG93bmxvYWQgUlN0dWRpb10oaHR0cDovL3d3dy5yc3R1ZGlvLmNvbS9pZGUpIGZvciBmcmVlIGJ5IGZvbGxvd2luZyB0aGUgaW5zdHJ1Y3Rpb25zIGF0IHRoZSBsaW5rLiBGb3Igbm9uLVIgdXNlcnMsIHdlIHJlY29tbWVuZCB0aGUgW0hhbmRzLW9uIFByb2dyYW1taW5nIHdpdGggUl0oaHR0cHM6Ly9yc3R1ZGlvLWVkdWNhdGlvbi5naXRodWIuaW8vaG9wci9wYWNrYWdlcy5odG1sKSBmb3IgYSBicmllZiBvdmVydmlldyBvZiB0aGUgc29mdHdhcmUncyBmdW5jdGlvbmFsaXR5LiBIZXJlYWZ0ZXIsIHdlIGFzc3VtZSB0aGF0IHRoZSByZWFkZXIgaGFzIGFuIGludHJvZHVjdG9yeSB1bmRlcnN0YW5kaW5nIG9mIHRoZSBSIHByb2dyYW1taW5nIGxhbmd1YWdlLg0KDQpJbiB0aGUgY29kZSBjaHVuayBiZWxvdywgd2UgbG9hZCB0aGUgcGFja2FnZXMgdXNlZCB0byBzdXBwb3J0IG91ciBhbmFseXNpcy4gTm90ZSB0aGF0IHRoZSBjb2RlIG9mIHRoaXMgYW5kIGFueSBvZiB0aGUgY29kZSBjaHVua3MgY2FuIGJlIGhpZGRlbiBieSBjbGlja2luZyBvbiB0aGUgJ0hpZGUnIGJ1dHRvbiB0byBmYWNpbGl0YXRlIHRoZSBuYXZpZ2F0aW9uLiAqKlRoZSByZWFkZXIgY2FuIGhpZGUgYWxsIGNvZGUgYW5kL29yIGRvd25sb2FkIHRoZSBSbWQgZmlsZSBhc3NvY2lhdGVkIHdpdGggdGhpcyBkb2N1bWVudCBieSBjbGlja2luZyBvbiB0aGUgQ29kZSBidXR0b24gb24gdGhlIHRvcCByaWdodCBjb3JuZXIgb2YgdGhpcyBkb2N1bWVudC4qKiBPdXIgaW5wdXQgYW5kIG91dHB1dCBmaWxlcyBjYW4gYWxzbyBiZSBhY2Nlc3NlZC8gZG93bmxvYWRlZCBmcm9tIFtmbWVnYWhlZC9jb3ZpZDE5LWRlYXRoc10oaHR0cHM6Ly9naXRodWIuY29tL2ZtZWdhaGVkL2NvdmlkMTktZGVhdGhzKS4gDQoNCg0KYGBge3IgcGFja2FnZXMsIGNhY2hlPUZBTFNFfQ0KIyBDaGVjayBhbmQgaW5zdGFsbCBpZiB0aGVzZSBwYWNrYWdlcyBhcmUgbm90IGZvdW5kIGxvY2FsbHkgb24gbWFjaGluZQ0KaWYocmVxdWlyZShwYWNtYW4pPT1GQUxTRSkgaW5zdGFsbC5wYWNrYWdlcygicGFjbWFuIikgIyBjaGVjayB0byBzZWUgaWYgdGhlIHBhY21hbiBwYWNrYWdlIGlzIGluc3RhbGxlZDsgaWYgbm90IGluc3RhbGwgaXQNCmlmKHJlcXVpcmUoZGV2dG9vbHMpPT1GQUxTRSkgaW5zdGFsbC5wYWNrYWdlcygiZGV2dG9vbHMiKSAjIGNoZWNrIHRvIHNlZSBpZiB0aGUgZGV2dG9vbHMgcGFja2FnZSBpcyBpbnN0YWxsZWQ7IGlmIG5vdCBpbnN0YWxsIGl0DQppZihyZXF1aXJlKHVyYm5tYXByKT09RkFMU0UpIGRldnRvb2xzOjppbnN0YWxsX2dpdGh1YignVXJiYW5JbnN0aXR1dGUvdXJibm1hcHInKQ0KaWYocmVxdWlyZShhbGJlcnN1c2EpPT1GQUxTRSkgZGV2dG9vbHM6Omluc3RhbGxfZ2l0aHViKCdocmJybXN0ci9hbGJlcnN1c2EnKSAjaW5zdGFsbCBwYWNrYWdlIGlmIG5lZWRlZA0KDQojIEluc3RhbGwgKGlmIG5lZWRlZCkgYW5kIGxvYWQgdGhlIGZvbGxvd2luZyBwYWNrYWdlcw0KcGFjbWFuOjpwX2xvYWQodGlkeXZlcnNlLCBtYWdyaXR0ciwgamFuaXRvciwgZGF0YVByZXBhcmF0aW9uLCBsdWJyaWRhdGUsIHNraW1yLCAjIGZvciBkYXRhIGFuYWx5c2lzDQogICAgICAgICAgICAgICBDT1ZJRDE5LCBydmVzdCwgIyBmb3IgZXh0cmFjdGluZyByZWxldmFudCBkYXRhDQogICAgICAgICAgICAgICBEVCwgcGFuZGVyLCBzdGFyZ2F6ZXIsIGtuaXRyLCAjIGZvciBmb3JtYXR0aW5nIGFuZCBuaWNlbHkgcHJpbnRlZCBvdXRwdXRzDQogICAgICAgICAgICAgICBzY2FsZXMsIFJDb2xvckJyZXdlciwgRGF0YUV4cGxvcmVyLCB0aWZmLCBncmlkLCMgZm9yIHBsb3RzDQogICAgICAgICAgICAgICBwbG90bHksIHVyYm5tYXByLCBhbGJlcnN1c2EsIHRpZ3JpcywgbGVhZmxldCwgdG1hcCwgIyBmb3IgbWFwcw0KICAgICAgICAgICAgICAgem9vLCBmcHAyLCBOYkNsdXN0LCAjIGZvciBUUyBhbmFseXNpcyBhbmQgY2x1c3RlcmluZw0KICAgICAgICAgICAgICAgVklNLCBubmV0LCBjYXJldCwgTXVNSW4sICMgZXhwbGFuYXRvcnkgbW9kZWxpbmcNCiAgICAgICAgICAgICAgIGNvbmZsaWN0ZWQpICMgZm9yIG1hbmFnaW5nIGNvbmZsaWN0cyBpbiBmdW5jdGlvbnMgd2l0aCBzYW1lIG5hbWVzDQoNCiMgSGFuZGxpbmcgY29uZmxpY3RpbmcgZnVuY3Rpb24gbmFtZXMgZnJvbSBwYWNrYWdlcw0KY29uZmxpY3RfcHJlZmVyKCdjb21iaW5lJywgJ2RwbHlyJykgIyBQcmVmZXJyaW5nIGRwbHlyOjpjb21iaW5lIG92ZXIgYW55IG90aGVyIHBhY2thZ2UNCmNvbmZsaWN0X3ByZWZlcignc2VsZWN0JywgImRwbHlyIikgIyBQcmVmZXJyaW5nIGRwbHlyOjpzZWxlY3Qgb3ZlciBhbnkgb3RoZXIgcGFja2FnZQ0KY29uZmxpY3RfcHJlZmVyKCJzdW1tYXJpemUiLCAiZHBseXIiKSAjIFByZWZlcnJpbmcgZHBseXI6OnN1bW1hcml6ZSBvdmVyIGFueSBvdGhlciBwYWNrYWdlDQpjb25mbGljdF9wcmVmZXIoImZpbHRlciIsICJkcGx5ciIpICMgUHJlZmVycmluZyBmaWx0ZXIgZnJvbSBkcGx5cg0KY29uZmxpY3RfcHJlZmVyKCJkaXN0IiwgInN0YXRzIikgIyBQcmVmZXJyaW5nIGRpc3QgZnJvbSBzdGF0cw0KY29uZmxpY3RfcHJlZmVyKCJhcy5kaXN0IiwgInN0YXRzIikgIyBQcmVmZXJyaW5nIGFzLmRpc3QgZnJvbSBzdGF0cw0KDQojIEN1c3RvbSBGdW5jdGlvbnMNCnNvdXJjZV91cmwoJ2h0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9mbWVnYWhlZC9jb3ZpZDE5LWRlYXRocy9tYXN0ZXIvTWFya2Rvd24vY3VzdG9tX2Z1bmN0aW9ucy5SJykNCg0Kc0luZm8gPSBzZXNzaW9uSW5mbygpICMgc2F2aW5nIGFsbCB0aGUgcGFja2FnZXMvZnVuY3Rpb25zIGFuZCBzZXNzaW9uIGluZm8NCmBgYA0KDQoNCiMgRXh0cmFjdGluZyB0aGUgRGF0YXNldHMNCg0KRm9yIG91ciBhbmFseXNpcywgd2UgZnVzZSBkYXRhIGZyb20gbXVsdGlwbGUgc291cmNlcy4gV2UgZGVzY3JpYmUgdGhlIHByb2Nlc3Mgb2Ygb2J0YWluaW5nIGFuZCBtZXJnaW5nIGVhY2ggb2YgdGhlc2Ugc291cmNlcyBpbiB0aGUgc3Vic2VjdGlvbnMgYmVsb3cuDQoNCiMjIFRpbWUgU2VyaWVzIERhdGENCg0KSW4gdGhpcyBzZWN0aW9uLCB3ZSB1dGlsaXplIHRoZSBbQ09WSUQxOSBwYWNrYWdlXShodHRwczovL2NyYW4uci1wcm9qZWN0Lm9yZy93ZWIvcGFja2FnZXMvQ09WSUQxOS9DT1ZJRDE5LnBkZikgdG8gb2J0YWluIHRoZSBmb2xsb3dpbmcgaW5mb3JtYXRpb246IFtAZ3VpZG90dGkyMDIwXSAgDQogIC0gKipDb25maXJtZWQgY2FzZXMsIHJlY292ZXJpZXMgYW5kIGRlYXRocyoqOyAgDQogIC0gKipQb2xpY3kgaW5mb3JtYXRpb24qKiAoZS5nLiwgdHJhbnNwb3J0IGNsb3NpbmcsIHNjaG9vbCBjbG9zaW5nLCBjbG9zaW5nIGV2ZW50LCBtb3ZlbWVudCByZXN0cmljdGlvbnMsIHRlc3RpbmcgcG9saWNpZXMsIGFuZCBjb250YWN0IHRyYWNpbmcpOyBhbmQgIA0KICAtICoqUG9wdWxhdGlvbiBhbmQgc3RhbmRhcmQgZ2VvZ3JhcGhpYyBpbmZvcm1hdGlvbioqIGZvciBlYWNoIGNvdW50eS4gIA0KICANCkZyb20gdGhpcyBpbmZvcm1hdGlvbiwgd2UgaGF2ZSBhbHNvIGNvbXB1dGVkIHRoZSBuZXcgZGFpbHkgYW5kIHdlZWtseSBjb25maXJtZWQgY2FzZXMvZGVhdGhzIHBlciBjb3VudHkuIFRoZSBkYXRhIGlzIHN0b3JlZCBpbiBhIHRpZHkgZm9ybWF0LCBidXQgY2FuIGJlIGV4cGFuZGVkIHRvIGEgd2lkZSBmb3JtYXQgdXNpbmcgYHBpdm90X3dpZGVyKClgIGZyb20gdGhlIFt0aWR5dmVyc2UgcGFja2FnZV0oaHR0cHM6Ly93d3cudGlkeXZlcnNlLm9yZy8pLg0KDQpgYGB7ciBjb3VudGllc30NCmVuZERhdGUgPSAnMjAyMi0wMS0yMicNCmVuZERhdGVQcmludFYgPSBmb3JtYXQoeW1kKGVuZERhdGUpLCBmb3JtYXQgPSAiJWIgJWQsICVZIikNCg0KY291bnRpZXMgPSBjb3ZpZDE5KA0KICBjb3VudHJ5ID0gIlVTIiwgDQogIGxldmVsID0gMywgIyBmb3IgY291bnR5DQogIHN0YXJ0ID0gIjIwMjAtMDMtMDEiLCAjIEZpcnN0IFN1bmRheSBpbiBNYXJjaA0KICBlbmQgPSBlbmREYXRlLCAjIGVuZCBEYXRlIA0KICByYXcgPSBGQUxTRSwgIyB0byBlbnN1cmUgdGhhdCBhbGwgY291bnRpZXMgaGF2ZSB0aGUgc2FtZSBncmlkIG9mIGRhdGVzDQogIGFtciA9IE5VTEwsICMgd2UgYXJlIG5vdCB1c2luZyB0aGUgYXBwbGUgbW9iaWxpdHkgZGF0YSBmb3Igb3VyIGFuYWx5c2lzDQogIGdtciA9IE5VTEwsICMgd2UgYXJlIG5vdCB1c2luZyB0aGUgR29vZ2xlIG1vYmlsaXR5IGRhdGEgZm9yIG91ciBhbmFseXNpcw0KICB3YiA9IE5VTEwsICMgd29ybGQgYmFuayBkYXRhIG5vdCBoZWxwZnVsIGZvciBjb3VudHkgbGV2ZWwgYW5hbHlzaXMNCiAgdmVyYm9zZSA9IEZBTFNFDQogICkNCg0KY291bnRpZXMgJTw+JSANCiAgIyBuZXh0IGxpbmUgcmVtb3ZlcyBub24tY29udGlndW91cyBVUyBzdGF0ZXMvdGVycml0b3JpZXMNCiAgZmlsdGVyKCFhZG1pbmlzdHJhdGl2ZV9hcmVhX2xldmVsXzIgJWluJSBjKCdBbGFza2EnLCAnSGF3YWlpJywgJ1B1ZXJ0byBSaWNvJywgJ05vcnRoZXJuIE1hcmlhbmEgSXNsYW5kcycsICdWaXJnaW4gSXNsYW5kcycpKSAlPiUgDQogICMgdGhlc2UgYXJlIG5vdCBjb3VudGllcw0KICBmaWx0ZXIoIWlzLm5hKGtleV9sb2NhbCkpICU+JSAgDQogICMgZ3JvdXBpbmcgdGhlIGRhdGEgYnkgdGhlIGlkIGNvbHVtbiB0byBtYWtlIGNvbXB1dGF0aW9ucyBjb3JyZWN0DQogIGdyb3VwX2J5KGlkKSAlPiUgDQogICMgdG8gZW5zdXJlIGNvcnJlY3QgY2FsY3VsYXRpb25zDQogIGFycmFuZ2UoaWQsIGRhdGUpICU+JSANCiAgbXV0YXRlKGRheSA9IHdkYXkoZGF0ZSwgbGFiZWwgPSBUUlVFKSAlPiUgZmFjdG9yKG9yZGVyZWQgPSBGKSwgIyBkYXkgb2Ygd2Vlaw0KICAgICAgICAgbmV3Q2FzZXMgPSBjKE5BLCBkaWZmKGNvbmZpcm1lZCkpLCAjICBuZXcgZGFpbHkgY2FzZXMgcGVyIGNvdW50eQ0KICAgICAgICAgbmV3RGVhdGhzID0gYyhOQSwgZGlmZihkZWF0aHMpKSApICAjIG5ldyBkYWlseSBkZWF0aHMgcGVyIGNvdW50eQ0KDQojIG1hbnVhbGx5IGlkZW50aWZ5aW5nIGZhY3RvciB2YXJpYWJsZXMNCmZhY3RvclZhcnMgPSBjKCJzY2hvb2xfY2xvc2luZyIsICJ3b3JrcGxhY2VfY2xvc2luZyIsICJjYW5jZWxfZXZlbnRzIiwNCiAgICAgICAgICAgICAgICJnYXRoZXJpbmdzX3Jlc3RyaWN0aW9ucyIsICJ0cmFuc3BvcnRfY2xvc2luZyIsICJzdGF5X2hvbWVfcmVzdHJpY3Rpb25zIiwNCiAgICAgICAgICAgICAgICJpbnRlcm5hbF9tb3ZlbWVudF9yZXN0cmljdGlvbnMiLCAiaW50ZXJuYXRpb25hbF9tb3ZlbWVudF9yZXN0cmljdGlvbnMiLA0KICAgICAgICAgICAgICAgImluZm9ybWF0aW9uX2NhbXBhaWducyIsICJ0ZXN0aW5nX3BvbGljeSIsICJjb250YWN0X3RyYWNpbmciKQ0KDQpjb3VudGllcyAlPD4lICMgY29udmVydGluZyB0aG9zZSB2YXJpYWJsZXMgaW50byBjaGFyYWN0ZXIgYW5kIHRoZW4gZmFjdG9yDQogIG11dGF0ZV9hdCgudmFycyA9IHZhcnMoYW55X29mKGZhY3RvclZhcnMpKSwgLmZ1bnMgPSBhcy5jaGFyYWN0ZXIpICU+JSANCiAgbXV0YXRlX2F0KC52YXJzID0gdmFycyhhbnlfb2YoZmFjdG9yVmFycykpLCAuZnVucyA9IGFzLmZhY3RvcikNCg0KDQojIHNhdmluZyB0aGUgcmVzdWx0cyBhcyBhIFJEUyBGaWxlDQpzYXZlUkRTKGNvdW50aWVzLCAnLi4vRGF0YS9PdXRwdXQvY291bnRpZXMucmRzJykNCmBgYA0KDQpBdCB0aGlzIHN0YWdlLCB3ZSBoYXZlIG9ubHkgcmVhZCB0aGUgZGF0YSBiYXNlZCBvbiB0aGUgY292aWQgcGFja2FnZS4gVGhlIHJlc3VsdGluZyBkYXRhIGlzIHN0b3JlZCBhdCBhbiBvYmplY3QgdGl0bGVkIGNvdW50aWVzLCB3aGljaCBjb250YWlucyBgciBucm93KGNvdW50aWVzKSAlPiUgY29tbWEoKWAgb2JzZXJ2YXRpb25zIGFuZCBgciBuY29sKGNvdW50aWVzKSAlPiUgY29tbWEoKWAgdmFyaWFibGVzLiBOb3RlIHRoYXQgd2UgaGF2ZSBmaWx0ZXJlZCBvYnNlcnZhdGlvbnMgdGhhdCBkbyBub3QgaGF2ZSBhIG51bWVyaWMga2V5IGFuZCByZW1vdmVkIHNvbWUgY29sdW1ucyB0aGF0IGRvIG5vdCBhZGQgYW55IHZhbHVlIHRvIGZ1dHVyZSBhbmFseXNpcyAoZS5nLiwgaW52YXJpYW50IGNvbHMpLg0KDQojIyBDcm9zcyBTZWN0aW9uYWwgRGF0YQ0KSW4gdGhpcyBhbmFseXNpcywgd2UgaGF2ZSBleHRyYWN0ZWQgdGhlIGZvbGxvd2luZyBkYXRhc2V0czogIA0KDQogICgxKSBUaGUgKipDREMncyBTb2NpYWwgVnVsbmVyYWJpbGl0eSBJbmRleCoqLCB3aGVyZSB3ZSB3b3VsZCB1dGlsaXplIHRoZSBmb2xsb3dpbmcgKmZvdXIgc3VtbWFyeSB0aGVtZSByYW5raW5nIHZhcmlhYmxlczoqICAgIA0KICAgIGEuICoqU29jaW9lY29ub21pYyoqIOKAkyBgUlBMX1RIRU1FMWAgIA0KICAgIGIuICoqSG91c2Vob2xkIENvbXBvc2l0aW9uICYgRGlzYWJpbGl0eSoqIOKAkyBgUlBMX1RIRU1FMmAgICANCiAgICBjLiAqKk1pbm9yaXR5IFN0YXR1cyAmIExhbmd1YWdlKiog4oCTIGBSUExfVEhFTUUzYCAgDQogICAgZC4gKipIb3VzaW5nIFR5cGUgJiBUcmFuc3BvcnRhdGlvbioqIOKAkyBgUlBMX1RIRU1FNGAgIA0KICAgIE5vdGUgdGhhdCBlYWNoIG9mIHRoZXNlIGluZGljZXMgd2VyZSBjb21wdXRlZCBieSB0aGUgW0NlbnRlcnMgZm9yIERpc2Vhc2UgQ29udHJvbCBhbmQgUHJldmVudGlvbi8gQWdlbmN5IGZvciBUb3hpYyBTdWJzdGFuY2VzIGFuZCBEaXNlYXNlIFJlZ2lzdHJ5LyBHZW9zcGF0aWFsIFJlc2VhcmNoLCBBbmFseXNpcywgYW5kIFNlcnZpY2VzIFByb2dyYW1dKGh0dHBzOi8vd3d3LmF0c2RyLmNkYy5nb3YvcGxhY2VhbmRoZWFsdGgvc3ZpL2RhdGFfZG9jdW1lbnRhdGlvbl9kb3dubG9hZC5odG1sKS4gVGhlIGNzdiBmaWxlIHRpdGxlZDogYFNWSTIwMThfVVNfQ09VTlRZLmNzdmAgd2FzIGRvd25sb2FkZWQgb24gTm92IDAyLCAyMDIwIChjbGljayBbaGVyZV0oaHR0cHM6Ly93d3cuYXRzZHIuY2RjLmdvdi9wbGFjZWFuZGhlYWx0aC9zdmkvZGF0YV9kb2N1bWVudGF0aW9uX2Rvd25sb2FkLmh0bWwpIHRvIGRvd25sb2FkIHRoZSBmaWxlIGFmdGVyIHNlbGVjdGluZyBgY291bnRpZXNgIGZvciB0aGUgKkdlb2dyYXBoeSBUeXBlKiBhbmQgYENTViBGaWxlYCBmb3IgdGhlICpGaWxlIHR5cGUqIGFuZCBjbGlja2luZyAqR08qKS4gIA0KICAgIA0KICAoMikgV2UgYWxzbyBleHRyYWN0ZWQgdGhlIGZvbGxvd2luZyAqKnBvbGl0aWNhbCBhbmQgYWRtaW5pc3RyYXRpdmUgdmFyaWFibGVzKiosIGFzIG9mIHRoZSBiZWdpbm5pbmcgb2YgdGhlIHBhbmRlbWljOiAgDQogICAgYS4gQmFzZWQgb24gQERWTi9WT1FDSFFfMjAyMSwgd2UgaGF2ZSBvYnRhaW5lZCB0aGUgdm90aW5nIHJlc3VsdHMgZm9yIGFsbCBjb3VudGllcyBpbiB0aGUgMjAxNiBQcmVzaWRlbnRpYWwgZWxlY3Rpb25zLiBUaGUgZGF0YSB3YXMgdXNlZCB0byBjb21wdXRlIHRoZSBwZXJjZW50YWdlIG9mIHRvdGFsIHZvdGVzIHRoYXQgd2VudCB0byBQcmVzaWRlbnQgVHJ1bXAsIHdpdGggdGhlIHVuZGVybHlpbmcgaHlwb3RoZXNpcyB0aGF0IHRoZSBwb2xpdGljaXphdGlvbiBvZiBDT1ZJRCByZXNwb25zZSAoZS5nLiwgcGVyY2VwdGlvbi93aWxsaW5nbmVzcyB0byB1c2UgZmFjZSBtYXNrcywgcG9saWNpZXMgYW5kIHRoZSBwb3B1bGF0aW9u4oCZcyByZWFjdGlvbiB0byB0aGUgZGlzZWFzZSkgbWF5IGJlIGV4cGxhaW5lZCBieSBwYXJ0eSBhZmZpbGlhdGlvbi5pbXB1dGVkIGl0cyB2YWx1ZSB3aXRoIOKAnERlbW9jcmF0aWPigJ0gc2luY2UgRC5DLuKAmXMgTWF5b3IgaXMgYSBEZW1vY3JhdCAoYW5kIERDIGlzIG5vdCBhIHN0YXRlKS4gIA0KICAgIGIuICoqU3RhdGUncyBDREMgUmVnaW9uIENsYXNzaWZpY2F0aW9uOioqIFdlIGhhdmUgYWxzbyBlbmdpbmVlcmVkIGEgcmVnaW9uIHZhcmlhYmxlIGJhc2VkIG9uIHRoZSBbQ0RD4oCZcyAxMCBSZWdpb25zIEZyYW1ld29ya10oaHR0cHM6Ly93d3cuY2RjLmdvdi9jb29yZGluYXRlZGNocm9uaWMvZG9jcy9uY2NkcGhwLXJlZ2lvbnMtbWFwLnBkZikuIFdoaWxlIGdlb2dyYXBoaWMgcmVnaW9ucyBhcmUgaHlwb3RoZXNpemVkIHRvIGJlIGEgZmFjdG9yIGluIGRpc2Vhc2Ugb3V0YnJlYWtzLCB3ZSBjaG9zZSB0byB1dGlsaXplIHRoZSBDREMgcmVnaW9ucyBzcGVjaWZpY2FsbHkgYmFzZWQgb24gdGhlIGZvbGxvd2luZyBleHBsYW5hdGlvbiBmcm9tIHRoZSBhZm9yZW1lbnRpb25lZCBsaW5rOiAq4oCcQ0RD4oCZcyBOYXRpb25hbCBDZW50ZXIgZm9yIENocm9uaWMgRGlzZWFzZSBQcmV2ZW50aW9uIGFuZCBIZWFsdGggUHJvbW90aW9uIChOQ0NEUEhQKSBpcyBzdHJlbmd0aGVuaW5nIHRoZSBjb25zaXN0ZW5jeSBhbmQgcXVhbGl0eSBvZiB0aGUgZ3VpZGFuY2UsIGNvbW11bmljYXRpb25zLCBhbmQgdGVjaG5pY2FsIGFzc2lzdGFuY2UgcHJvdmlkZWQgdG8gc3RhdGVzIHRvIGltcHJvdmUgY29vcmRpbmF0aW9uIGFjcm9zcyBvdXIgc3RhdGUgcHJvZ3JhbXPigJ0qICANCiAgDQogICgzKSBXZSBhbHNvIGV4dHJhY3RlZCAqKmFuIG92ZXJhbGwgZ292ZXJubWVudCByZXNwb25zZSBpbmRleCBjYXB0dXJpbmcgdGhlIHN0cmVuZ3RoIG9mIENPVklELTE5IHJlc3BvbnNlIHBvbGljaWVzIG9uIGEgc3RhdGUgKGFuZCB0aGUgRGlzdHJpY3Qgb2YgQ29sdW1iaWEpIGxldmVsKiogZnJvbSB0aGUgW0JsYXZhdG5payBTY2hvb2wgb2YgR292ZXJubWVudCdzIEdpdEh1YiBSZXBvc2l0b3J5XShodHRwczovL2dpdGh1Yi5jb20vT3hDR1JUL1VTQS1jb3ZpZC1wb2xpY3kpLiBUaGlzIGluZGV4IGNhcHR1cmVzIDEzIGRpZmZlcmVudCBpbmRpY2F0b3JzLCBjYXB0dXJpbmcgdGhlIGBgZnVsbCByYW5nZSBvZiBnb3Zlcm5tZW50IHJlc3BvbnNlJycuIERldGFpbHMgZm9yIGhvdyB0aGlzIGluZGljYXRvciBpcyBjb21wdXRlZCBjYW4gYmUgZm91bmQgYXQgW0JTRy1XUC0yMDIwLzAzNF0oaHR0cHM6Ly93d3cuYnNnLm94LmFjLnVrL3NpdGVzL2RlZmF1bHQvZmlsZXMvMjAyMC0wOC9CU0ctV1AtMjAyMC0wMzQucGRmKS4NCg0KYGBge3IgZXh0cmFjdGluZ0RhdGF9DQojICgxKSBPYnRhaW5pbmcgdGhlIHN2aSBkYXRhDQpzdmkgPC0gcmVhZF9jc3YoImh0dHBzOi8vcmF3LmdpdGh1YnVzZXJjb250ZW50LmNvbS9mbWVnYWhlZC9jb3ZpZDE5LWRlYXRocy9tYXN0ZXIvRGF0YS9JbnB1dC9TVkkyMDE4X1VTX0NPVU5UWS5jc3YiKSAlPiUgDQogIGNsZWFuX25hbWVzKCkgJT4lICMgbWFrZSBldmVyeXRoaW5nIGxvd2VyX2Nhc2UNCiAgc2VsZWN0KGxvY2F0aW9uLCBmaXBzLCBzdGF0ZSwgY291bnR5LCBhcmVhX3NxbWksIGVfdG90cG9wLA0KICAgICAgICAgcnBsX3RoZW1lMSwgcnBsX3RoZW1lMiwgcnBsX3RoZW1lMywgcnBsX3RoZW1lNCkgJT4lICMgY29sdW1ucyBvZiBpbnRlcmVzdA0KICBtdXRhdGUoZV9wb3BkZW5zaXR5ID0gZV90b3Rwb3AvIGFyZWFfc3FtaSkgJT4lICMgY29tcHV0aW5nIHBvcHVsYXRpb24gZGVuc2l0eQ0KICBmaWx0ZXIoYWNyb3NzKHdoZXJlKGlzLm51bWVyaWMpLCB+LiA+PSAwKSkgJT4lICAjIGJlY2F1c2UgTkEncyBhcmUgY29kZWQgYXMgLTk5OQ0KICBmaWx0ZXIoIXN0YXRlICVpbiUgYygnSEFXQUlJJywgJ0FMQVNLQScpKSAjIGZpbHRlcmluZyB0byBjb250aWd1b3VzIFVTDQoNCg0KIyAoMikgUG9saXRpY2FsIGFuZCBBZG1pbmlzdHJhdGl2ZSBEYXRhDQojIyMjIChhKSBDb3VudHkgdm90ZSBpbiB0aGUgMjAyMCBwcmVzaWRlbnRpYWwgZWxlY3Rpb25zDQplbGVjdGlvbnMgPSByZWFkLmNzdigiLi4vRGF0YS9JbnB1dC9jb3VudHlwcmVzXzIwMDAtMjAyMC5jc3YiKSAlPiUgIyByZWFkaW5nIHRoZSBkb3dubG9hZGVkIENTVg0KICBmaWx0ZXIoeWVhciA9PSAyMDIwICYgcGFydHkgPT0gIlJFUFVCTElDQU4iKSAlPiUgIyBqdXN0IGtlZXBpbmcgZGF0YSBmb3IgcmVjZW50IGVsZWN0aW9uIGFuZCByZXB1YmxpY2FuIHZvdGVzDQogIG11dGF0ZShrZXlfbG9jYWwgPSBhcy5jaGFyYWN0ZXIoY291bnR5X2ZpcHMpICU+JSANCiAgICAgICAgICAgc3RyX3BhZCh3aWR0aCA9IDUsIHNpZGUgPSAnbGVmdCcsIHBhZCA9ICcwJykpICU+JQ0KICBncm91cF9ieShrZXlfbG9jYWwpICU+JSANCiAgc3VtbWFyaXNlKA0KICAgICMgKyBhYnNlbnRlZSwgYWR2YW5jZWQgdm90aW5nLCBwcm92LCBlbGVjdGlvbiBkYXkgYW5kIGFic2VudGVlIGJ5IG1haWwgdm90ZXMNCiAgICBjYW5kaWRhdGV2b3RlcyA9IHN1bShjYW5kaWRhdGV2b3RlcyksDQogICAgIyAgdGFraW5nIGFueSBvZiB0aGVzZSB2YWx1ZXMgZm9yIHRvdGFsIHZvdGVzIChpdCBkb2VzIG5vdCBjaGFuZ2UgYnkgdHlwZSkNCiAgICB0b3RhbHZvdGVzID0gdG90YWx2b3RlcywNCiAgICAjIGNhbGN1bGF0aW5nIHRvdGFsIHZvdGVzIHJlY2VpdmVkIGJ5IHRoZSBjYW5kaWRhdGUNCiAgICBwZXJjUmVwVm90ZXMgPSAxMDAqKGNhbmRpZGF0ZXZvdGVzL3RvdGFsdm90ZXMpICkgJT4lIA0KIyBjb21wdXRpbmcgcGVyY2VudCBvZiByZXB1YmxpY2FuIHZvdGVzIChmcm9tIHRvdGFsIHZvdGVzKQ0KICBzZWxlY3Qoa2V5X2xvY2FsLCBwZXJjUmVwVm90ZXMpICU+JSAjIGtlZXBpbmcgb25seSB0aGUga2V5IGFuZCB2YXJpYWJsZSB1c2VkIGluIG1lcmdlDQogIHVuaXF1ZSgpDQoNCg0KIyMjIyAoYikgU3RhdGUgUmVnaW9uIENsYXNzaWZpY2F0aW9uDQpjZGNSZWdpb25zID0gZGF0YS5mcmFtZShzdGF0ZSA9IGMoJ0Nvbm5lY3RpY3V0JywgJ01haW5lJywgJ01hc3NhY2h1c2V0dHMnLCAnTmV3IEhhbXBzaGlyZScsICdSaG9kZSBJc2xhbmQnICwgDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdWZXJtb250JywgJ05ldyBZb3JrJywgIyBFbmQgb2YgUmVnaW9uIEENCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJ0RlbGF3YXJlJywgJ0Rpc3RyaWN0IG9mIENvbHVtYmlhJywgJ01hcnlsYW5kJywgJ1Blbm5zeWx2YW5pYScsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdWaXJnaW5pYScsICdXZXN0IFZpcmdpbmlhJywgJ05ldyBKZXJzZXknLCAjIEVuZCBvZiBSZWdpb24gQg0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAnTm9ydGggQ2Fyb2xpbmEnLCAnU291dGggQ2Fyb2xpbmEnLCAnR2VvcmdpYScsICdGbG9yaWRhJywgIyBSZWdpb24gQw0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAnS2VudHVja3knLCAnVGVubmVzc2VlJywgJ0FsYWJhbWEnLCAnTWlzc2lzc2lwcGknLCAjIFJlZ2lvbiBEDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdJbGxpbm9pcycsICdJbmRpYW5hJywgJ01pY2hpZ2FuJywgJ01pbm5lc290YScsICdPaGlvJywNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJ1dpc2NvbnNpbicsICMgRW5kIG9mIFJlZ2lvbiBFDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdBcmthbnNhcycsICdMb3Vpc2lhbmEnLCAnTmV3IE1leGljbycsICdPa2xhaG9tYScsICdUZXhhcycsICMgUmVnaW9uIEYNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJ0lvd2EnLCAnS2Fuc2FzJywgJ01pc3NvdXJpJywgJ05lYnJhc2thJywgIyBSZWdpb24gRw0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAnQ29sb3JhZG8nLCAnTW9udGFuYScsICdOb3J0aCBEYWtvdGEnLCAnU291dGggRGFrb3RhJywNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJ1V0YWgnLCAnV3lvbWluZycsICMgRW5kIG9mIFJlZ2lvbiBIDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICdBcml6b25hJywgJ0NhbGlmb3JuaWEnLCAnSGF3YWlpJywgJ05ldmFkYScsICMgUmVnaW9uIEkNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgJ0FsYXNrYScsICdJZGFobycsICdPcmVnb24nLCAnV2FzaGluZ3RvbicgIyBSZWdpb24gSg0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICApICU+JSB0b3VwcGVyKCksDQogICAgICAgICAgICAgICAgICAgICAgICByZWdpb24gPSBjKHJlcCgnQScsIDcpLCByZXAoJ0InLCA3KSwgcmVwKCdDJywgNCksDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICByZXAoJ0QnLCA0KSwgcmVwKCdFJywgNiksIHJlcCgnRicsIDUpLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgcmVwKCdHJywgNCksIHJlcCgnSCcsIDYpLCByZXAoJ0knLCA0KSwNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIHJlcCgnSicsIDQpICkgKQ0KDQojICgzKSBQb2xpY3kgRGF0YQ0KcG9saWN5IDwtIHJlYWRfY3N2KCdodHRwczovL3Jhdy5naXRodWJ1c2VyY29udGVudC5jb20vT3hDR1JUL1VTQS1jb3ZpZC1wb2xpY3kvbWFzdGVyL2RhdGEvT3hDR1JUX1VTX2xhdGVzdC5jc3YnKQ0KcG9saWN5IDwtIGZpbHRlcihwb2xpY3ksICFpcy5uYShSZWdpb25OYW1lKSB8ICFSZWdpb25OYW1lICVpbiUgYygnQWxhc2thJywgJ0hhd2FpaScpKQ0KcG9saWN5JHN0YXRlIDwtIHRvdXBwZXIocG9saWN5JFJlZ2lvbk5hbWUpICMgYSBzdGF0ZSB2YXJpYWJsZSA9IGFuIHVwcGVyIGNhc2Ugb2YgZXhpc3RpbmcgUmVnaW9uTmFtZQ0KcG9saWN5JERhdGUgJTw+JSB5bWQoKSAjIGNvbnZlcnRpbmcgdGhlIERhdGUgZGF0YSB0byBhIGRhdGUgZm9ybWF0DQoNCiMgQ2FsY3VsYXRpbmcgYSBzdW1tYXJ5IHRhYmxlIG9mIG1lZGlhbiB2YWx1ZSBmb3IgdGhlIEdvdmVybm1lbnRSZXNwb25zZUluZGV4IHBlciBzdGF0ZQ0KcG9saWN5U3VtbWFyeSA8LSBwb2xpY3kgJT4lDQogIGZpbHRlcihEYXRlID49ICcyMDIwLTAzLTAxJyAmIERhdGUgPD0gZW5kRGF0ZSkgJT4lICMgdG8gbWF0Y2ggb3VyIENPVklEIERhdGEgdGltZVNlcmllcw0KICBncm91cF9ieShzdGF0ZSkgJT4lICMgcGVyZm9ybSBjb21wdXRhdGlvbnMgdXNpbmcgdGhlIG1lZGlhbiB2YWx1ZSwgcGVyIHN0YXRlLCBmb3IgZWFjaCBpbmRleA0KICBzdW1tYXJpc2UoR292ZXJubWVudFJlc3BvbnNlSW5kZXhNZWRpYW4gPSBtZWRpYW4oR292ZXJubWVudFJlc3BvbnNlSW5kZXgsIG5hLnJtID0gVFJVRSkpDQoNCiMgZWRpdGluZyB0aGUgbmFtZSBvZiBXYXNoaW5ndG9uIERDIHRvIERpc3RyaWN0IG9mIENvbHVtYmlhIHRvIG1hdGNoIHRoZSBvdGhlciBkYXRhc2V0cw0KcG9saWN5U3VtbWFyeSRzdGF0ZSAlPD4lICBzdHJfcmVwbGFjZSgnV0FTSElOR1RPTiBEQycsICdESVNUUklDVCBPRiBDT0xVTUJJQScpDQoNCg0KIyAoNCkgQ29tYmluaW5nIGFsbCB0aGUgcG90ZW50aWFsIHByZWRpY3RvcnMgaW4gYSBjcm9zc1NlY3Rpb25EYXRhbCBGcmFtZQ0KY3Jvc3NTZWN0aW9uYWxEYXRhIDwtIGxlZnRfam9pbihzdmksIGNkY1JlZ2lvbnMsIGJ5ID0gJ3N0YXRlJykgJT4lDQogIGxlZnRfam9pbihwb2xpY3lTdW1tYXJ5LCBieSA9ICdzdGF0ZScpICU+JSANCiAgcmVuYW1lKGtleV9sb2NhbCA9IGZpcHMpDQoNCmNyb3NzU2VjdGlvbmFsRGF0YSA8LSBsZWZ0X2pvaW4oY3Jvc3NTZWN0aW9uYWxEYXRhLCBlbGVjdGlvbnMsDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgIGJ5ID0gJ2tleV9sb2NhbCcpDQoNCiMgc2F2aW5nIHRoZSByZXN1bHRzIGFzIGEgUkRTIEZpbGUNCnNhdmVSRFMoY3Jvc3NTZWN0aW9uYWxEYXRhLCAnLi4vRGF0YS9PdXRwdXQvY3Jvc3NTZWN0aW9uYWxEYXRhLnJkcycpDQoNCiMgVGFidWxhdGluZyB0aGUgcmVzdWx0cyBhbmQgcHJvdmlkaW5nIGEgd2F5IHRvIGV4cG9ydCB0aGUgdGFibGUgdG8gZGlmZmVyZW50IGZvcm1hdHMNCmRhdGF0YWJsZShjcm9zc1NlY3Rpb25hbERhdGEgJT4lIHNlbGVjdCgtYyhzdGF0ZSwgY291bnR5KSksDQogICAgICAgICAgZXh0ZW5zaW9ucyA9IGMoJ0ZpeGVkQ29sdW1ucycsICdCdXR0b25zJyksIG9wdGlvbnMgPSBsaXN0KA0KICAgICAgICAgICAgZG9tID0gJ0JmcnRpcCcsDQogICAgICAgICAgICBzY3JvbGxYID0gVFJVRSwNCiAgICAgICAgICAgIGJ1dHRvbnMgPSBjKCdjb3B5JywgJ2NzdicsICdleGNlbCcsICdwZGYnKSwNCiAgICAgICAgICAgIGZpeGVkQ29sdW1ucyA9IGxpc3QobGVmdENvbHVtbnMgPSAyKSkNCiAgICAgICAgICApICU+JSANCiAgZm9ybWF0Um91bmQoY29sdW1ucz0gYygnYXJlYV9zcW1pJywgJ3JwbF90aGVtZTEnLCAncnBsX3RoZW1lMicsDQogICAgICAgICAgICAgICAgICAgICAgICAgJ3JwbF90aGVtZTMnLCAncnBsX3RoZW1lNCcsICdlX3BvcGRlbnNpdHknKSwNCiAgICAgICAgICAgICAgZGlnaXRzPTIpDQpgYGANCg0KIyMgRXhwbG9yYXRvcnkgQW5hbHlzaXMNCg0KSW4gdGhpcyBzZWN0aW9uLCB3ZSBwZXJmb3JtIGFuIGV4cGxvcmF0b3J5IGFuYWx5c2lzIG9uIHRoZSBkYXRhIG9idGFpbmVkIGZyb20gdGhlIG11bHRpcGxlIHNvdXJjZXMuIA0KDQojIyMgQ3VtdWxhdGl2ZSBDYXNlcw0KDQpgYGB7ciBjdW1DYXNlcywgZmlnLnNob3c9J2xhc3QnfQ0Kbm9Hb29nbGVOQXMgPC0gZmlsdGVyKGNvdW50aWVzLCAhaXMubmEoa2V5X2dvb2dsZV9tb2JpbGl0eSkpICMgcmVtb3ZpbmcgTkFzIGZyb20ga2V5X2dvb2dsZV9tb2JpbGl0eQ0KaWRJbmRleCA8LSBzYW1wbGUobm9Hb29nbGVOQXMkaWQsIDkpICMgc2FtcGxpbmcgOSBjb3VudGllcyBieSBpZA0KDQojIFNhdmluZyBjdW11bGF0aXZlIGRlYXRocyBmaWd1cmUgdG8gYW4gdGlmZiBmaWxlDQp0aWZmKGZpbGVuYW1lID0gJy4uL0ZpZ3VyZXMvc2FtcGxlQ3VtdWxhdGl2ZUNhc2VzLnRpZmYnLA0KICAgIHdpZHRoID0gMTM2NiwgaGVpZ2h0ID03NjgsIHBvaW50c2l6ZSA9IDE2KQ0KY291bnRpZXMgJT4lIGZpbHRlcihpZCAlaW4lIGlkSW5kZXgpICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IGNvbmZpcm1lZCwgZ3JvdXAgPSBpZCwgY29sb3IgPSBrZXlfZ29vZ2xlX21vYmlsaXR5KSkgKw0KICBnZW9tX2xpbmUoc2l6ZSA9IDEuMjUpICsNCiAgc2NhbGVfeF9kYXRlKGRhdGVfYnJlYWtzID0gIjEgbW9udGgiLCBkYXRlX2xhYmVscyA9ICIlYiIpICsNCiAgZmFjZXRfd3JhcCh+IGtleV9nb29nbGVfbW9iaWxpdHksIHNjYWxlcyA9ICdmcmVlX3knLCBuY29sID0gMykgKw0KICB0aGVtZShsZWdlbmQucG9zaXRpb24gPSAnbm9uZScpICsgDQogIGxhYnMoY29sb3IgPSAnJywgeCA9ICdNb250aCcsIHkgPSAnQ3VtdWxhdGl2ZSBDYXNlcyBCeSBDb3VudHknKSArDQogIHNjYWxlX2NvbG9yX2JyZXdlcih0eXBlID0gJ3F1YWwnLCBwYWxldHRlID0gJ1BhaXJlZCcpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KIyBDcmVhdGluZyBhbiBpbnRlcmFjdGl2ZSBwbG90IGZvciB0aGUgbWFya2Rvd24NCnAgPC0gZ2dwbG90Mjo6bGFzdF9wbG90KCkgKyBnZW9tX2xpbmUoc2l6ZSA9IDAuNzUpICsgIyBtb2RpZnlpbmcgdGhlIHBsb3QgZm9yIHBsb3RseQ0KICB0aGVtZV9idyhiYXNlX3NpemUgPSA5KSArIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICdub25lJykgIyB0byBtYWtlIG1hcmdpbnMgc21hbGxlcg0KZ2dwbG90bHkocCwgaGVpZ2h0ID0gNzY4KSAlPiUgIGxheW91dF9nZ3Bsb3RseSgpDQpgYGANCg0KIyMjIEN1bXVsYXRpdmUgRGVhdGhzDQoNCmBgYHtyIGN1bURlYXRocywgZmlnLnNob3c9J2xhc3QnfQ0KIyBTYXZpbmcgY3VtdWxhdGl2ZSBkZWF0aHMgZmlndXJlIHRvIGFuIHRpZmYgZmlsZQ0KdGlmZihmaWxlbmFtZSA9ICcuLi9GaWd1cmVzL3NhbXBsZUN1bXVsYXRpdmVEZWF0aHMudGlmZicsDQogICAgd2lkdGggPSAxMzY2LCBoZWlnaHQgPTc2OCwgcG9pbnRzaXplID0gMTYpDQpjb3VudGllcyAlPiUgZmlsdGVyKGlkICVpbiUgaWRJbmRleCkgJT4lIA0KICBnZ3Bsb3QoYWVzKHggPSBkYXRlLCB5ID0gZGVhdGhzLCBncm91cCA9IGlkLCBjb2xvciA9IGtleV9nb29nbGVfbW9iaWxpdHkpKSArDQogIGdlb21fbGluZShzaXplID0gMS4yNSkgKw0KICBzY2FsZV94X2RhdGUoZGF0ZV9icmVha3MgPSAiMSBtb250aCIsIGRhdGVfbGFiZWxzID0gIiViIikgKw0KICBmYWNldF93cmFwKH4ga2V5X2dvb2dsZV9tb2JpbGl0eSwgc2NhbGVzID0gJ2ZyZWVfeScsIG5jb2wgPSAzKSArDQogIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICdub25lJykgKyANCiAgbGFicyhjb2xvciA9ICcnLCB4ID0gJ01vbnRoJywgeSA9ICdDdW11bGF0aXZlIERlYXRocyBCeSBDb3VudHknKSArDQogIHNjYWxlX2NvbG9yX2JyZXdlcih0eXBlID0gJ3F1YWwnLCBwYWxldHRlID0gJ1BhaXJlZCcpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KIyBDcmVhdGluZyBhbiBpbnRlcmFjdGl2ZSBwbG90IGZvciB0aGUgbWFya2Rvd24NCnAgPC0gZ2dwbG90Mjo6bGFzdF9wbG90KCkgKyBnZW9tX2xpbmUoc2l6ZSA9IDAuNzUpICsgIyBtb2RpZnlpbmcgdGhlIHBsb3QgZm9yIHBsb3RseQ0KICB0aGVtZV9idyhiYXNlX3NpemUgPSA5KSArIHRoZW1lKGxlZ2VuZC5wb3NpdGlvbiA9ICdub25lJykgIyB0byBtYWtlIG1hcmdpbnMgc21hbGxlcg0KZ2dwbG90bHkocCwgaGVpZ2h0ID0gNzY4KSAlPiUgIGxheW91dF9nZ3Bsb3RseSgpDQpgYGANCg0KIyMjIFJQTCBUaGVtZSAxDQpgYGB7ciBycGwxTWFwfQ0KIyBSZXRyaWV2aW5nIHRoZSBVLlMuIGNvdW50eSBjb21wb3NpdGUgbWFwIGFzIGEgc2ltcGxlZmVhdHVyZQ0KY3R5X3NmID0gZ2V0X3VyYm5fbWFwKG1hcCA9ICJjb3VudGllcyIsIHNmID0gVFJVRSkgJT4lIGZpbHRlcighc3RhdGVfbmFtZSAlaW4lIGMoJ0FsYXNrYScsICdIYXdhaWknKSkNCmN0eV9zZiAlPD4lIGdlb19qb2luKGNyb3NzU2VjdGlvbmFsRGF0YSwgYnlfc3A9ICdjb3VudHlfZmlwcycsIGJ5X2RmPSAna2V5X2xvY2FsJykNCg0KIyBTYXZpbmcgYSBzdGF0aWMgdmVyc2lvbiBvZiB0aGUgZmlndXJlIGFzIHRpZmYNCnRpZmYoZmlsZW5hbWUgPSAnLi4vRmlndXJlcy9ycGxUaGVtZTFNYXAudGlmZicsIHdpZHRoID0gMTM2NiwgaGVpZ2h0ID03NjgsIHBvaW50c2l6ZSA9IDE2KQ0KdG1fc2hhcGUoY3R5X3NmKSArIHRtX3BvbHlnb25zKCdycGxfdGhlbWUxJywgdGl0bGUgPSAnUlBMIFRoZW1lIDE6IFNvY2lvZWNvbm9taWMnLCBwYWxldHRlID0gIi1HcmVlbnMiKQ0KaW52aXNpYmxlKCBkZXYub2ZmKCkgKSAjIHRvIHN1cHByZXNzIHRoZSB1bndhbnRlZCBvdXRwdXQgZnJvbSBkZXYub2ZmDQoNCiMgUHJpbnRpbmcgYSBwbmcgdmVyc2lvbiBvZiB0aGUgcGxvdCBpbiBNYXJrZG93biAobG93ZXIgcXVhbGl0eSBpbWFnZSBmb3IgcXVpY2tlciBjb21waWxhdGlvbiBvZiBIVE1MKQ0KcmVhZFRJRkYoIi4uL0ZpZ3VyZXMvcnBsVGhlbWUxTWFwLnRpZmYiKSAlPiUgZ3JpZC5yYXN0ZXIoKQ0KYGBgDQoNCiMjIyBSUEwgVGhlbWUgMg0KYGBge3IgcnBsMk1hcH0NCiMgU2F2aW5nIGEgc3RhdGljIHZlcnNpb24gb2YgdGhlIGZpZ3VyZSAoY2FwaXRhbGl6aW5nIG9uIHRoZSB0bWFwIHBhY2thZ2UpDQp0aWZmKGZpbGVuYW1lID0gJy4uL0ZpZ3VyZXMvcnBsVGhlbWUyTWFwLnRpZmYnLCB3aWR0aCA9IDEzNjYsIGhlaWdodCA9NzY4LCBwb2ludHNpemUgPSAxNikNCnRtX3NoYXBlKGN0eV9zZikgKyB0bV9wb2x5Z29ucygncnBsX3RoZW1lMicsIHRpdGxlID0gJ1JQTCBUaGVtZSAyOiBIb3VzZWhvbGQgQ29tcG9zaXRpb24gJiBEaXNhYmlsaXR5JywgcGFsZXR0ZSA9ICItR3JlZW5zIikNCmludmlzaWJsZSggZGV2Lm9mZigpICkgIyB0byBzdXBwcmVzcyB0aGUgdW53YW50ZWQgb3V0cHV0IGZyb20gZGV2Lm9mZg0KDQojIFByaW50aW5nIGEgcG5nIHZlcnNpb24gb2YgdGhlIHBsb3QgaW4gTWFya2Rvd24gKGxvd2VyIHF1YWxpdHkgaW1hZ2UgZm9yIHF1aWNrZXIgY29tcGlsYXRpb24gb2YgSFRNTCkNCnJlYWRUSUZGKCIuLi9GaWd1cmVzL3JwbFRoZW1lMk1hcC50aWZmIikgJT4lIGdyaWQucmFzdGVyKCkNCmBgYA0KDQojIyMgUlBMIFRoZW1lIDMNCmBgYHtyIHJwbDNNYXB9DQojIFNhdmluZyBhIHN0YXRpYyB2ZXJzaW9uIG9mIHRoZSBmaWd1cmUgKGNhcGl0YWxpemluZyBvbiB0aGUgdG1hcCBwYWNrYWdlKQ0KdGlmZihmaWxlbmFtZSA9ICcuLi9GaWd1cmVzL3JwbFRoZW1lM01hcC50aWZmJywgd2lkdGggPSAxMzY2LCBoZWlnaHQgPTc2OCwgcG9pbnRzaXplID0gMTYpDQp0bV9zaGFwZShjdHlfc2YpICsgdG1fcG9seWdvbnMoJ3JwbF90aGVtZTMnLCB0aXRsZSA9ICdSUEwgVGhlbWUgMzogTWlub3JpdHkgU3RhdHVzICYgTGFuZ3VhZ2UnLCBwYWxldHRlID0gIi1HcmVlbnMiKQ0KaW52aXNpYmxlKCBkZXYub2ZmKCkgKSAjIHRvIHN1cHByZXNzIHRoZSB1bndhbnRlZCBvdXRwdXQgZnJvbSBkZXYub2ZmDQoNCiMgUHJpbnRpbmcgYSBwbmcgdmVyc2lvbiBvZiB0aGUgcGxvdCBpbiBNYXJrZG93biAobG93ZXIgcXVhbGl0eSBpbWFnZSBmb3IgcXVpY2tlciBjb21waWxhdGlvbiBvZiBIVE1MKQ0KcmVhZFRJRkYoIi4uL0ZpZ3VyZXMvcnBsVGhlbWUzTWFwLnRpZmYiKSAlPiUgZ3JpZC5yYXN0ZXIoKQ0KYGBgDQoNCiMjIyBSUEwgVGhlbWUgNA0KYGBge3IgcnBsNE1hcH0NCiMgU2F2aW5nIGEgc3RhdGljIHZlcnNpb24gb2YgdGhlIGZpZ3VyZSAoY2FwaXRhbGl6aW5nIG9uIHRoZSB0bWFwIHBhY2thZ2UpDQp0aWZmKGZpbGVuYW1lID0gJy4uL0ZpZ3VyZXMvcnBsVGhlbWU0TWFwLnRpZmYnLCB3aWR0aCA9IDEzNjYsIGhlaWdodCA9NzY4LCBwb2ludHNpemUgPSAxNikNCnRtX3NoYXBlKGN0eV9zZikgKyB0bV9wb2x5Z29ucygncnBsX3RoZW1lNCcsIHRpdGxlID0gJ1JQTCBUaGVtZSA0OiBIb3VzaW5nIFR5cGUgJiBUcmFuc3BvcnRhdGlvbicsIHBhbGV0dGUgPSAiLUdyZWVucyIpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KIyBQcmludGluZyBhIHBuZyB2ZXJzaW9uIG9mIHRoZSBwbG90IGluIE1hcmtkb3duIChsb3dlciBxdWFsaXR5IGltYWdlIGZvciBxdWlja2VyIGNvbXBpbGF0aW9uIG9mIEhUTUwpDQpyZWFkVElGRigiLi4vRmlndXJlcy9ycGxUaGVtZTRNYXAudGlmZiIpICU+JSBncmlkLnJhc3RlcigpDQpgYGANCg0KIyMjIENEQyBSZWdpb25zDQpgYGB7ciBjZGNNYXB9DQpjcm9zc1NlY3Rpb25hbERhdGEkc3RhdGUgJTw+JSB0b2xvd2VyKCkgJT4lIHRvb2xzOjp0b1RpdGxlQ2FzZSgpDQojIFJldHJpZXZpbmcgdGhlIFUuUy4gc3RhdGUgY29tcG9zaXRlIG1hcCBhcyBhIHNpbXBsZWZlYXR1cmUNCnN0YXRlX3NmID0gZ2V0X3VyYm5fbWFwKG1hcCA9ICJzdGF0ZXMiLCBzZiA9IFRSVUUpICU+JSANCiAgZmlsdGVyKCFzdGF0ZV9uYW1lICVpbiUgYygnQWxhc2thJywgJ0hhd2FpaScpICkNCnN0YXRlX3NmICU8PiUgZ2VvX2pvaW4oY3Jvc3NTZWN0aW9uYWxEYXRhLCBieV9zcD0gJ3N0YXRlX25hbWUnLCBieV9kZj0gJ3N0YXRlJykNCg0KIyBTYXZpbmcgYSBzdGF0aWMgdmVyc2lvbiBvZiB0aGUgZmlndXJlIChjYXBpdGFsaXppbmcgb24gdGhlIHRtYXAgcGFja2FnZSkNCnRpZmYoZmlsZW5hbWUgPSAnLi4vRmlndXJlcy9jZGNNYXAudGlmZicsIHdpZHRoID0gMTM2NiwgaGVpZ2h0ID03NjgsIHBvaW50c2l6ZSA9IDE2KQ0KdG1fc2hhcGUoc3RhdGVfc2YpICsgdG1fcG9seWdvbnMoJ3JlZ2lvbicsIHRpdGxlID0gJ0NEQyBSZWdpb24nLCBwYWxldHRlID0gIlBhaXJlZCIpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KIyBQcmludGluZyBhIHBuZyB2ZXJzaW9uIG9mIHRoZSBwbG90IGluIE1hcmtkb3duIChsb3dlciBxdWFsaXR5IGltYWdlIGZvciBxdWlja2VyIGNvbXBpbGF0aW9uIG9mIEhUTUwpDQpyZWFkVElGRigiLi4vRmlndXJlcy9jZGNNYXAudGlmZiIpICU+JSBncmlkLnJhc3RlcigpDQpgYGANCg0KIyMjIFJlcHVibGljYW4gVm90ZXMgaW4gdGhlIDIwMjAgUHJlc2lkZW50aWFsIEVsZWN0aW9ucw0KYGBge3IgZ292TWFwfQ0KIyBTYXZpbmcgYSBzdGF0aWMgdmVyc2lvbiBvZiB0aGUgZmlndXJlIChjYXBpdGFsaXppbmcgb24gdGhlIHRtYXAgcGFja2FnZSkNCnRpZmYoZmlsZW5hbWUgPSAnLi4vRmlndXJlcy9nb3Zlcm5vcnNNYXAudGlmZicsIHdpZHRoID0gMTM2NiwgaGVpZ2h0ID03NjgsIHBvaW50c2l6ZSA9IDE2KQ0KdG1fc2hhcGUoc3RhdGVfc2YpICsgDQogIHRtX3BvbHlnb25zKCdwZXJjUmVwVm90ZXMnLCB0aXRsZSA9ICIyMDIwIFJlcHVibGljYW4gVm90ZXMiLCBwYWxldHRlID0gJ1JlZHMnKQ0KaW52aXNpYmxlKCBkZXYub2ZmKCkgKSAjIHRvIHN1cHByZXNzIHRoZSB1bndhbnRlZCBvdXRwdXQgZnJvbSBkZXYub2ZmDQoNCiMgUHJpbnRpbmcgYSBwbmcgdmVyc2lvbiBvZiB0aGUgcGxvdCBpbiBNYXJrZG93biAobG93ZXIgcXVhbGl0eSBpbWFnZSBmb3IgcXVpY2tlciBjb21waWxhdGlvbiBvZiBIVE1MKQ0KcmVhZFRJRkYoIi4uL0ZpZ3VyZXMvZ292ZXJub3JzTWFwLnRpZmYiKSAlPiUgZ3JpZC5yYXN0ZXIoKQ0KYGBgDQoNCiMjIyBHb3Zlcm5tZW50IFJlc3BvbnNlIEluZGV4IE1lZGlhbiBWYWx1ZQ0KYGBge3IgZ292UmVzcG9uc2VNYXB9DQojIFNhdmluZyBhIHN0YXRpYyB2ZXJzaW9uIG9mIHRoZSBmaWd1cmUgKGNhcGl0YWxpemluZyBvbiB0aGUgdG1hcCBwYWNrYWdlKQ0KdGlmZihmaWxlbmFtZSA9ICcuLi9GaWd1cmVzL2dvdmVybm1lbnRSZXNwb25zZU1hcC50aWZmJywgd2lkdGggPSAxMzY2LCBoZWlnaHQgPTc2OCwgcG9pbnRzaXplID0gMTYpDQp0bV9zaGFwZShzdGF0ZV9zZikgKyB0bV9wb2x5Z29ucygnR292ZXJubWVudFJlc3BvbnNlSW5kZXhNZWRpYW4nLCB0aXRsZSA9ICJNZWRpYW4gVmFsdWUgb2YgdGhlIEdvdmVybm1lbnQgUmVzcG9uc2UgSW5kZXgiLCBwYWxldHRlID0gIi1HcmVlbnMiKQ0KaW52aXNpYmxlKCBkZXYub2ZmKCkgKSAjIHRvIHN1cHByZXNzIHRoZSB1bndhbnRlZCBvdXRwdXQgZnJvbSBkZXYub2ZmDQoNCiMgUHJpbnRpbmcgYSBwbmcgdmVyc2lvbiBvZiB0aGUgcGxvdCBpbiBNYXJrZG93biAobG93ZXIgcXVhbGl0eSBpbWFnZSBmb3IgcXVpY2tlciBjb21waWxhdGlvbiBvZiBIVE1MKQ0KcmVhZFRJRkYoIi4uL0ZpZ3VyZXMvZ292ZXJubWVudFJlc3BvbnNlTWFwLnRpZmYiKSAlPiUgZ3JpZC5yYXN0ZXIoKQ0KYGBgDQoNCg0KIyBUaW1lLVNlcmllcyBDbHVzdGVyaW5nIA0KDQpJdCBpcyBpbXBvcnRhbnQgdG8gbm90ZSB0aGF0LCBpbiBvdXIgZXN0aW1hdGlvbiwgdGhlcmUgYXJlIHRocmVlIGltcG9ydGFudCBkZWNpc2lvbnMgdG8gYmUgbWFkZSB3aGVuIHBlcmZvcm1pbmcgdGltZS1zZXJpZXMgY2x1c3RlcmluZzogIA0KDQogIC0gKlByZXBhcmF0aW9uIG9mIHRoZSBEaWZmZXJlbnQgVGltZS1TZXJpZXMgdG8gYmUgQ2x1c3RlcmVkKiAgSW4gdGhpcyBzZWN0aW9uLCB3ZSBoYXZlIChhKSBzZWxlY3RlZCB0aGUgbmV3IGRhaWx5IGRlYXRocyBwZXIgY291bnR5IGFzIHRoZSBwcmltYXJ5IHZhcmlhYmxlIG9mIGludGVyZXN0LCAoYikgc21vb3RoZWQgdGhhdCB2YXJpYWJsZSB1c2luZyBhIHNldmVuLWRheSBtb3ZpbmcgYXZlcmFnZSwgYW5kIChjKSBzY2FsZWQgdGhlIG9ic2VydmF0aW9ucyB3aXRoaW4gZWFjaCBjb3VudHnigJlzIDctZGF5IE1BIG9mIG5ldyBkYWlseSBkZWF0aHMgc3VjaCB0aGF0IGl0IGlzIGJvdW5kZWQgYmV0d2VlbiAwIGFuZCAxLiBUaGlzIGFsbG93cyB1cyB0byBjb21wYXJlIHRoZSAqKnNoYXBlKiogb2YgdGhlIHRpbWUtc2VyaWVzL3Byb2ZpbGUgYWNyb3NzIGNvdW50aWVzIG9mIGRpZmZlcmVudCBwb3B1bGF0aW9ucyBhbmQgd2hlcmUgdGhlIG1hZ25pdHVkZSBvZiB0aGUgZGVhdGhzIGlzIHF1aXRlIGRpZmZlcmVudC4NCiAgDQogIC0gKkNob2ljZSBvZiBEaXN0YW5jZSBNZWFzdXJlOiBUaGUgRXVjbGlkZWFuIGRpc3RhbmNlKiwgVGhlIEV1Y2xpZGVhbiBEaXN0YW5jZSBpLmUuLCB0aGUgJGxfMiQgbm9ybSwgaXMgdGhlIG1vc3QgY29tbW9ubHkgdXNlZCBkaXN0YW5jZSBtZWFzdXJlIHNpbmNlIGl0IGlzIGNvbXB1dGF0aW9uYWxseSBlZmZpY2llbnQuIEhvd2V2ZXIsIGl0IG1heSBub3QgYmUgc3VpdGFibGUgZm9yIGFwcGxpY2F0aW9ucyB3aGVyZSB0aGUgdGltZS1zZXJpZXMgYXJlIG9mIGRpZmZlcmVudCBsZW5ndGggaW4gYWRkaXRpb24gdG8gYmVpbmcgc2Vuc2l0aXZlIHRvIG5vaXNlLCBzY2FsZSBhbmQgdGltZSBzaGlmdHMgKFNhcmTDoS1Fc3Bpbm9zYSwgMjAxNykuICANCiAgDQogIC0gKkNob2ljZSBvZiBDbHVzdGVyaW5nIEFsZ29yaXRobToqIEEgbGFyZ2UgbnVtYmVyIG9mIGNsdXN0ZXJpbmcgYWxnb3JpdGhtcyBoYXZlIGJlZW4gcHJvcG9zZWQgaW4gdGhlIGxpdGVyYXR1cmUuIE1vc3QgY29tbW9uIGNsdXN0ZXJpbmcgYXBwcm9hY2hlcyBhcmUgc2hhcGUtYmFzZWQsIHdoaWNoIGluY2x1ZGUgJGstJG1lYW5zIGNsdXN0ZXJpbmcgYW5kIGhpZXJhcmNoaWNhbCBjbHVzdGVyaW5nLiBUaGUgcmVhZGVyIGlzIHJlZmVycmVkIHRvIEBhZ2hhYm96b3JnaTIwMTV0aW1lIGZvciBhIGRldGFpbGVkIHJldmlldy4gSW4gb3VyIHByZWxpbWluYXJ5IGFuYWx5c2lzLCB3ZSBoYXZlIGNob3NlbiB0byB1c2UgdGhlIGhpZXJhcmNoaWNhbCBjbHVzdGVyaW5nIGFwcHJvYWNoIHNpbmNlIGl0IHByb3ZpZGVzIGFuIGVhc3kgdG8gdW5kZXJzdGFuZCBkZW5kb2dyYW0gYW5kIHRoZSBudW1iZXIgb2YgY291bnRpZXMgd2FzIHNtYWxsLiBIb3dldmVyLCBpbiBvdXIgZnVsbCBhbmFseXNpcywgd2Ugd2lsbCB1c2UgdGhlICRrLSRtZWFucyBjbHVzdGVyaW5nIGFsZ29yaXRobSBzaW5jZSBpdCBpcyBjb21wdXRhdGlvbmFsbHkgZWZmaWNpZW50LiBGdXJ0aGVybW9yZSwgd2Ugb3ZlcmNhbWUgdGhlIHRyYWRpdGlvbmFsIGxpbWl0YXRpb24gb2YgaGF2aW5nIHRvIHByZS1zcGVjaWZ5ICRrJCBieSB1dGlsaXppbmcgMjYgaW5kZXhlcyBmb3IgZGV0ZXJtaW5pbmcgdGhlIG9wdGltYWwgbnVtYmVyIG9mIGNsdXN0ZXJzIGluIGEgZGF0YSBzZXQgYmFzZWQgb24gdGhlIGV4Y2VsbGVudCBhcHByb2FjaCBhbmQgcGFja2FnZSBpbXBsZW1lbnRhdGlvbiBvZiBAY2hhcnJhZDIwMTROYkNsdXN0Lg0KDQoNCiMjIERhdGEgUHJlcGFyYXRpb24NCg0KYGBge3IgZGF0YVByZXBDbHVzdGVyaW5nfQ0KY2x1c3RlcmluZ1ByZXAgPC0gY291bnRpZXMgJT4lICMgZnJvbSB0aGUgY291bnRpZXMNCiAgc2VsZWN0KGlkLCBkYXRlLCBrZXlfbG9jYWwsIG5ld0RlYXRocykgJT4lICMgc2VsZWN0aW5nIG1pbmltYWwgYW1vdW50IG9mIGNvbHMgZm9yIHZpc3VhbCBpbnNwZWN0aW9uDQogIGFycmFuZ2UoaWQsIGRhdGUpICU+JSAjIGFycmFuZ2VkIHRvIGVuc3VyZSBjb3JyZWN0IGNhbGN1bGF0aW9ucw0KICBtdXRhdGUobmV3TUE3ID0gcm9sbG1lYW5yKG5ld0RlYXRocywgayA9IDcsIGZpbGwgPSBOQSksICMgNy1kYXkgbW0gb2YgbmV3IChhZGp1c3RlZCkgZGVhdGhzDQogICAgICAgICBtYXhNQTcgPSBtYXgobmV3TUE3LCBuYS5ybSA9IFQpLCAjIG9idGFpbmluZyB0aGUgbWF4IHBlciBjb3VudHkgdG8gc2NhbGUgZGF0YQ0KICAgICAgICAgc2NhbGVkTmV3TUE3ID0gcG1heCgwLCBuZXdNQTcvbWF4TUE3LCBuYS5ybSA9IFRSVUUpICkgJT4lICMgc2NhbGluZyBkYXRhIHRvIGEgMC0xIHNjYWxlIGJ5IGNvdW50eQ0KICBzZWxlY3QoaWQsIGtleV9sb2NhbCwgZGF0ZSwgc2NhbGVkTmV3TUE3KSAlPiUgIyBkcm9wcGluZyB0aGUgdmFyaWFibGUgbmV3RGVhdGhzDQogIHBpdm90X3dpZGVyKG5hbWVzX2Zyb20gPSBkYXRlLCB2YWx1ZXNfZnJvbSA9IHNjYWxlZE5ld01BNykgIyBjb252ZXJ0aW5nIHRoZSBkYXRhIHRvIGEgd2lkZSBmb3JtYXQgZm9yIGNsdXN0ZXJpbmcNCg0KY29uc3RhbnRDb2x1bW5zICA8LSB3aGljaF9hcmVfY29uc3RhbnQoY2x1c3RlcmluZ1ByZXAsIHZlcmJvc2UgPSBGKSAjIGlkZW50aWZ5aW5nIGNvbnN0YW50IGNvbHVtbnMNCmRhdGVzRHJvcHBlZCA8LSBjb2xuYW1lcyhjbHVzdGVyaW5nUHJlcClbY29uc3RhbnRDb2x1bW5zXSAjIHVzZWQgZm9yIHByaW50aW5nIHRoZSBuYW1lcyBhZnRlciB0aGUgY29kZSBjaHVuaw0KDQpjbHVzdGVyaW5nUHJlcCAlPD4lIHNlbGVjdCgtYWxsX29mKGNvbnN0YW50Q29sdW1ucykgKSAlPiUgICMgc3BlZWRzIHVwIGNsdXN0ZXJpbmcgYnkgZGVjIGxlbmd0aCBvZiBzZXJpZXMNCiAgYXMuZGF0YS5mcmFtZSgpICMgZGF0YSBuZWVkcyB0byBiZSBkYXRhIGZyYW1lIGZvciBjbHVzdGVyaW5nDQpyb3cubmFtZXMoY2x1c3RlcmluZ1ByZXApID0gY2x1c3RlcmluZ1ByZXBbLDFdICMgbmVlZGVkIGZvciB0c2NsdXN0DQoNCmNsdXN0ZXJpbmdQcmVwID0gY2x1c3RlcmluZ1ByZXBbLC1jKDEpXSAlPiUgDQogIHNlbGVjdF9pZih+ICFhbnkoaXMubmEoLikpKSAjIHJlbW92aW5nIGFueSBkYXRlIGNvbnRhaW5pbmcgTkENCg0KIyBzYXZpbmcgdGhlIHJlc3VsdHMgYXMgYSBSRFMgRmlsZQ0Kc2F2ZVJEUyhjbHVzdGVyaW5nUHJlcCwgJy4uL0RhdGEvT3V0cHV0L2NsdXN0ZXJpbmdQcmVwLnJkcycpDQpgYGANCg0KVGhlIGZvbGxvd2luZyBkYXRlcyB3ZXJlIHJlbW92ZWQgZnJvbSBvdXIgZGF0YSBmcmFtZSBzaW5jZSB0aGUgYHNjYWxlZE5FV01BN2AgdmFyaWFibGUgd2FzIGNvbnN0YW50IGFjcm9zcyBhbGwgY291bnRpZXM6IGByIHByaW50KGRhdGVzRHJvcHBlZCwgc2VwID0iLCAiKWAuDQoNCg0KIyMgQ2x1c3RlcmluZyBDb250aWd1b3VzIFUuUy4gQ291bnRpZXMNCg0KYGBge3IgdHNDbHVzdGVyaW5nLCBmaWcuc2hvdz0naGlkZSd9DQpjbHVzdGVyaW5nUHJlcCAlPD4lIHNlbGVjdCgtYyhrZXlfbG9jYWwpKSAgIyByZW1vdmluZyB0aGlzIHZhcmlhYmxlIHNvIHdlIGNhbiBjbHVzdGVyDQoNCm5jICA8LSBOYkNsdXN0KGNsdXN0ZXJpbmdQcmVwLCBkaXN0YW5jZSA9ICJldWNsaWRlYW4iLCAjIGV1Y2xpZGVhbiBkaXN0YW5jZQ0KICAgICAgICAgICAgIG1pbi5uYyA9IDIsIG1heC5uYyA9IDUxLCAjIHNlYXJjaGluZyBmb3Igb3B0aW1hbCBrIGJldHdlZW4gaz0yIGFuZCBrPTUxDQogICAgICAgICAgICAgbWV0aG9kID0gImttZWFucyIsICMgdXNpbmcgdGhlIGstbWVhbnMgbWV0aG9kDQogICAgICAgICAgICAgaW5kZXggPSAiYWxsIikgIyB1c2luZyAyNiBvZiB0aGUgMzAgaW5kaWNlcyBpbiB0aGUgcGFja2FnZQ0KDQprY2x1cyAgPC0gbmMkQmVzdC5wYXJ0aXRpb24gJT4lIGFzLmRhdGEuZnJhbWUoKSAlPiUgI29idGFpbmluZyB0aGUgYmVzdCBwYXJ0aXRpb24vIGNsdXN0ZXIgYXNzaWdubWVudCBmb3Igb3B0aW1hbCBrDQogIHJlbmFtZSguLCBjbHVzdGVyX2dyb3VwID0gLikgJT4lIHJvd25hbWVzX3RvX2NvbHVtbigiQ291bnR5IikgDQoNCiNjb252ZXJ0aW5nIHRoZSB3aWRlIHRvIHRhbGwgZGF0YSBhbmQgYWRkaW5nIHRoZSBjbHVzdGVyIGdyb3VwaW5ncw0KY2x1c3RlcnMgIDwtIGNsdXN0ZXJpbmdQcmVwICU+JSANCiAgcm93bmFtZXNfdG9fY29sdW1uKHZhciA9ICJDb3VudHkiKSAlPiUgDQogIHBpdm90X2xvbmdlcihjb2xzID0gc3RhcnRzX3dpdGgoIjIwMjAiKSwgbmFtZXNfdG8gPSAiRGF0ZSIpICU+JSANCiAgaW5uZXJfam9pbiguLCBrY2x1cywgYnkgPSAiQ291bnR5IikgJT4lIA0KICBtdXRhdGUoY2x1c3Rlcl9ncm91cCA9IGFzLmZhY3RvcihjbHVzdGVyX2dyb3VwKSkNCg0KaWRDbHVzdGVycyAgPC0gY2x1c3RlcnMgJT4lIHNlbGVjdChjKENvdW50eSwgY2x1c3Rlcl9ncm91cCkpICMgY3JlYXRpbmcgYSBsb29rLXVwIHRhYmxlIG9mIGNvdW50eSBhbmQgY2x1c3RlciBncm91cA0KY29sbmFtZXMoaWRDbHVzdGVycykgIDwtIGMoJ2lkJywgJ2NsdXN0ZXJfZ3JvdXAnKSAjIHJlbmFtaW5nIHRoZSBjb2x1bW5zDQppZENsdXN0ZXJzICU8PiUgIHVuaXF1ZSgpICNyZW1vdmluZyB0aGUgZHVwbGljYXRlcyBkdWUgdG8gZGlmZmVyZW50IGRhdGVzICh3ZSBoYWQgdGhhdCB0byBlbnN1cmUgdGhhdCB0aGUgY2x1c3RlcmluZyB3YXMgYXBwbGllZCBjb3JyZWN0bHkpDQoNCiMgQWRkaW5nIENsdXN0ZXIgR3JvdXBpbmcgdG8gYSBzdWJzZXQgb2YgdGhlIGNvdW50aWVzIGRhdGEgZnJhbWUNCmNsdXN0ZXJDb3VudGllcyA8LSBjb3VudGllcyAlPiUgDQogIHNlbGVjdChjKGlkLCBrZXlfbG9jYWwsIGtleV9nb29nbGVfbW9iaWxpdHksIGFkbWluaXN0cmF0aXZlX2FyZWFfbGV2ZWxfMiwgYWRtaW5pc3RyYXRpdmVfYXJlYV9sZXZlbF8zKSkgJT4lIA0KICBpbm5lcl9qb2luKC4sIGlkQ2x1c3RlcnMsIGJ5ID0naWQnKSAlPiUgDQogIG11dGF0ZShjbHVzdGVyX2dyb3VwID0gcGFzdGUwKCdDJywgY2x1c3Rlcl9ncm91cCkpICU+JSANCiAgdW5pcXVlKCkNCg0KIyBzYXZpbmcgdGhlIHJlc3VsdHMgYXMgYSBSRFMgRmlsZQ0Kc2F2ZVJEUyhjbHVzdGVyQ291bnRpZXMsICcuLi9EYXRhL091dHB1dC9jbHVzdGVyQ291bnRpZXMucmRzJykNCmBgYA0KDQoNCiMjIFZpc3VhbGl6aW5nIHRoZSBDbHVzdGVyaW5nIFJlc3VsdHMNCg0KSW4gdGhpcyBzdWJzZWN0aW9uLCB3ZSBwcm92aWRlIHRocmVlIHBsb3RzOiAgDQoNCiAgLSBBIHBhbmVsZWQgc3BhZ2hldHRpIHBsb3QsIGhpZ2hsaWdodGluZyB0aGUgbWVkaWFuIHNjYWxlZCB0aW1lLXNlcmllcyBmb3IgcHJvZmlsZSBmb3IgZWFjaCBjbHVzdGVyOyAgDQogIC0gQSBwYW5lbCBwbG90IHdoZXJlIHRoZSBmaXJzdCwgc2Vjb25kIGFuZCB0aGlyZCBxdWFydGlsZXMgb2YgdGhlIHNjYWxlZCB0aW1lLXNlcmllcyBmb3IgZWFjaCBjbHVzdGVyIGFyZSBjb21wYXJlZDsgYW5kICANCiAgLSBBbiBpbnRlcmFjdGl2ZSBjaGxvcm9wbGV0aCBtYXBzIHRvIHZpc3VhbGl6ZSB0aGUgc3BhdGlhbCBkaXN0cmlidXRpb24gb2YgdGhlIGNsdXN0ZXJzLCB3aGVyZSB0aGUgcmVhZGVyIGNhbiBjbGljayBvbiBhIGdpdmVuIGNvdW50eSB0byBzaG93OiAoYSkgY291bnR5IG5hbWUsIChiKSBhc3NpZ25lZCBjbHVzdGVyLCAoYykgcG9wdWxhdGlvbiBkZW5zaXR5LCBhbmQgKGQpIHBlcmNlbnRhZ2Ugb2YgcmVzaWRlbnRzIGluIHBvdmVydHkuDQoNCg0KIyMjIFNwYWdoZXR0aSBQbG90DQoNCmBgYHtyIHNwYWdoZXR0aX0NCnNwYWdoZXR0aURGIDwtIGNvdW50aWVzICU+JSAjIGZyb20gdGhlIGNvdW50aWVzDQogIHNlbGVjdChpZCwgZGF0ZSwgbmV3RGVhdGhzLCBrZXlfZ29vZ2xlX21vYmlsaXR5KSAlPiUgIyBzZWxlY3RpbmcgbWluaW1hbCBjb2x1bW5zDQogIGxlZnRfam9pbihjbHVzdGVyQ291bnRpZXNbLCBjKCdpZCcsICdjbHVzdGVyX2dyb3VwJyldLCBieSA9ICdpZCcpICU+JSAjIHRvIGdldCBjbHVzdGVycw0KICBhcnJhbmdlKGlkLCBkYXRlKSAlPiUgIyBhcnJhbmdlZCB0byBlbnN1cmUgY29ycmVjdCBjYWxjdWxhdGlvbnMNCiAgbXV0YXRlKG5ld01BNyA9IHJvbGxtZWFucihuZXdEZWF0aHMsIGsgPSA3LCBmaWxsID0gTkEpLCAjIDctZGF5IG1hIG9mIG5ldyAoYWRqdXN0ZWQpIGRlYXRocw0KICAgICAgICAgbWF4TUE3ID0gbWF4KG5ld01BNywgbmEucm0gPSBUKSwgIyBvYnRhaW5pbmcgdGhlIG1heCBwZXIgY291bnR5IHRvIHNjYWxlIGRhdGENCiAgICAgICAgIHNjYWxlZE5ld01BNyA9IHBtYXgoMCwgbmV3TUE3L21heE1BNywgbmEucm0gPSBUUlVFKSApICU+JSANCiAgdW5ncm91cCgpICU+JSBzZWxlY3QoZGF0ZSwgY2x1c3Rlcl9ncm91cCwgc2NhbGVkTmV3TUE3LCBrZXlfZ29vZ2xlX21vYmlsaXR5KSAlPiUgDQogIGdyb3VwX2J5KGRhdGUsIGNsdXN0ZXJfZ3JvdXApDQoNCnNwYWdoZXR0aURGJGNsdXN0ZXJfZ3JvdXAgJTw+JSBhcy5mYWN0b3IoKSANCg0KIyBDcmVhdGluZyBhIE5hbWVkIENvbG9yIFNjYWxlDQpjb2xvclBhbCA8LSAgYnJld2VyLnBhbChuPSBsZXZlbHMoc3BhZ2hldHRpREYkY2x1c3Rlcl9ncm91cCkgJT4lIGxlbmd0aCgpLCAnU2V0MicpDQpuYW1lcyhjb2xvclBhbCkgPC0gbGV2ZWxzKHNwYWdoZXR0aURGJGNsdXN0ZXJfZ3JvdXApDQoNCiMgU2F2aW5nIHNwYWdoZXR0aSBwbG90IHRvIGFuIHRpZmYgZmlsZQ0KdGlmZihmaWxlbmFtZSA9ICcuLi9GaWd1cmVzL3NwYWdoZXR0aVBsb3QudGlmZicsIHdpZHRoID0gMTM2NiwgaGVpZ2h0ID03NjgsIHBvaW50c2l6ZSA9IDE2KQ0Kc3BhZ2hldHRpREYgJT4lICANCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IHNjYWxlZE5ld01BNywgY29sb3IgPSBjbHVzdGVyX2dyb3VwLCBncm91cCA9IGtleV9nb29nbGVfbW9iaWxpdHkpKSArDQogIHNjYWxlX3hfZGF0ZShkYXRlX2JyZWFrcyA9ICIxIG1vbnRoIiwgZGF0ZV9sYWJlbHMgPSAiJWIiKSArDQogIGdlb21fbGluZShzaXplID0gMC4yNSwgYWxwaGEgPSAwLjEpICsNCiAgc3RhdF9zdW1tYXJ5KGFlcyhncm91cCA9IDEpLCANCiAgICAgICAgICAgICAgIGZ1bj0gbWVkaWFuLA0KICAgICAgICAgICAgICAgZ2VvbSA9ICJsaW5lIiwNCiAgICAgICAgICAgICAgIHNpemUgPSAxLjI1LCBjb2wgPSAnYmxhY2snKSArIA0KICBmYWNldF93cmFwKH4gY2x1c3Rlcl9ncm91cCwgbmNvbCA9IDEpICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gJ25vbmUnKSArIA0KICBsYWJzKHggPSAnTW9udGgnLCB5ID0gJ1NjYWxlZCBOZXcgRGVhdGhzIEJ5IENsdXN0ZXIgQnkgRGF5JywNCiAgICAgICBjYXB0aW9uID0gcGFzdGUwKCdTb2xpZCBibGFjayBsaW5lIHJlcHJlc2VudHMgdGhlIG1lZGlhbiBmb3IgZWFjaCBjbHVzdGVyIHwgDQogICAgICAgQmFzZWQgb24gRGF0YSBmcm9tIE1hcmNoIDAxLCAyMDIwIC0gJywgZW5kRGF0ZVByaW50VikgKSAgKw0KICBzY2FsZV9jb2xvcl9tYW51YWwodmFsdWVzID0gY29sb3JQYWwpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KIyBQcmludGluZyBhIHBuZyB2ZXJzaW9uIG9mIHRoZSBwbG90IGluIE1hcmtkb3duIChsb3dlciBxdWFsaXR5IGltYWdlIGZvciBxdWlja2VyIGNvbXBpbGF0aW9uIG9mIEhUTUwpDQpyZWFkVElGRigiLi4vRmlndXJlcy9zcGFnaGV0dGlQbG90LnRpZmYiKSAlPiUgZ3JpZC5yYXN0ZXIoKQ0KYGBgDQoNCg0KIyMjIFN1bW1hcnkgUGxvdA0KYGBge3Igc3VtbWFyeVBsb3R9DQojIENyZWF0aW5nIGEgZGF0YSBmcmFtZSBjb250YWluaW5nIHN0YXRpc3RpY2FsIHN1bW1hcmllcyBvZiB0aGUgdGltZSBzZXJpZXMgYnkgY2x1c3Rlcl9ncm91cA0Kc3VtbWFyeURmIDwtIHNwYWdoZXR0aURGICU+JSANCiAgc3VtbWFyaXNlKE1lZGlhbiA9IG1lZGlhbihzY2FsZWROZXdNQTcsIG5hLnJtPSBUUlVFKSwNCiAgICAgICAgICAgIGBGaXJzdCBRdWFydGlsZWAgPSBxdWFudGlsZShzY2FsZWROZXdNQTcsIHByb2JzID0gMC4yNSwgbmEucm09IFRSVUUpLA0KICAgICAgICAgICAgYFRoaXJkIFF1YXJ0aWxlYCA9IHF1YW50aWxlKHNjYWxlZE5ld01BNywgcHJvYnMgPSAwLjc1LCBuYS5ybT0gVFJVRSkpICU+JSANCiAgcGl2b3RfbG9uZ2VyKGNvbHMgPSBjKGBGaXJzdCBRdWFydGlsZWAsIE1lZGlhbiwgYFRoaXJkIFF1YXJ0aWxlYCksDQogICAgICAgICAgICAgICAgICAgICAgICBuYW1lc190byA9ICdTdGF0aXN0aWMnKQ0KDQp0aWZmKGZpbGVuYW1lID0gJy4uL0ZpZ3VyZXMvc3VtbWFyeVBsb3QudGlmZicsIHdpZHRoID0gMTM2NiwgaGVpZ2h0ID03NjgsIHBvaW50c2l6ZSA9IDE2KQ0Kc3VtbWFyeURmICU+JSANCiAgZ2dwbG90KGFlcyh4ID0gZGF0ZSwgeSA9IHZhbHVlLCBjb2xvciA9IGNsdXN0ZXJfZ3JvdXAsIGxpbmV0eXBlID0gIFN0YXRpc3RpYykpICsNCiAgc2NhbGVfeF9kYXRlKGRhdGVfYnJlYWtzID0gIjEgbW9udGgiLCBkYXRlX2xhYmVscyA9ICIlYiIpICsNCiAgZ2VvbV9saW5lKHNpemUgPSAxLjI1KSArDQogICBzY2FsZV9saW5ldHlwZV9tYW51YWwodmFsdWVzID0gYygnZG90dGVkJywgJ3NvbGlkJywgJ3R3b2Rhc2gnKSkgKw0KICBmYWNldF93cmFwKH4gY2x1c3Rlcl9ncm91cCwgbmNvbCA9IDEpICsNCiAgdGhlbWUobGVnZW5kLnBvc2l0aW9uID0gJ3RvcCcpICsgDQogIGxhYnMoY29sb3IgPSAnJywgeCA9ICdNb250aCcsIHkgPSAnUXVhcnRpbGVzIG9mIFNjYWxlZCBOZXcgRGVhdGhzIEJ5IENsdXN0ZXIgQnkgRGF5JykgKw0KICBzY2FsZV9jb2xvcl9tYW51YWwodmFsdWVzID0gY29sb3JQYWwpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KIyBQcmludGluZyBhIHBuZyB2ZXJzaW9uIG9mIHRoZSBwbG90IGluIE1hcmtkb3duIChsb3dlciBxdWFsaXR5IGltYWdlIGZvciBxdWlja2VyIGNvbXBpbGF0aW9uIG9mIEhUTUwpDQpyZWFkVElGRigiLi4vRmlndXJlcy9zdW1tYXJ5UGxvdC50aWZmIikgJT4lIGdyaWQucmFzdGVyKCkNCmBgYA0KDQojIyMgQ2x1c3RlciBNYXANCg0KYGBge3IgY2x1c3Rlck1hcCwgb3V0LndpZHRoPScxMDAlJ30NCiMgSm9pbmluZyB0aGUgY2x1c3RlckNvdW50aWVzIHJlc3VsdHMgd2l0aCB0aGUgZXhpc3RpbmcgY291bnR5IHNpbXBsZSBmZWF0dXJlcyBvYmplY3QgKGN0eV9zZikNCmNsdXN0ZXJDb3VudGllcyAlPD4lIHVuZ3JvdXAoKQ0KY3R5X3NmICU8PiUgbGVmdF9qb2luKGNsdXN0ZXJDb3VudGllc1ssIGMoJ2tleV9sb2NhbCcsICdjbHVzdGVyX2dyb3VwJyldLCBieSA9IGMoJ2NvdW50eV9maXBzJz0gJ2tleV9sb2NhbCcpKSAjIGFkZGluZyBjbHVzdGVyX2dyb3VwIHRvIGN0eV9zZg0KDQojIENyZWF0aW5nIGEgc3RhdGljIHZpc3VhbCBmb3IgdGhlIHBhcGVyDQp0aWZmKGZpbGVuYW1lID0gJy4uL0ZpZ3VyZXMvY2x1c3Rlck1hcC50aWZmJywgd2lkdGggPSAxMzY2LCBoZWlnaHQgPTc2OCwgcG9pbnRzaXplID0gMTYpDQp0bV9zaGFwZShjdHlfc2YpICsgdG1fcG9seWdvbnMoJ2NsdXN0ZXJfZ3JvdXAnLCB0aXRsZSA9ICdDbHVzdGVyICMnLCBwYWxldHRlID0gY29sb3JQYWwpDQppbnZpc2libGUoIGRldi5vZmYoKSApICMgdG8gc3VwcHJlc3MgdGhlIHVud2FudGVkIG91dHB1dCBmcm9tIGRldi5vZmYNCg0KDQojIENyZWF0aW5nIGFuIGludGVyYWN0aXZlIHZpc3VhbCBVc2luZyB0aGUgTGVhZmxldCBQYWNrYWdlDQojIyMjIENyZWF0aW5nIGEgbG9uZ2xhdCBwcm9qZWN0aW9uIChyZXF1aXJlZCBieSBsZWFmbGV0KQ0KbGVhZmxldF9zZiA8LSBjb3VudGllc19zZigibG9uZ2xhdCIpICU+JSBmaWx0ZXIoIXN0YXRlICVpbiUgYygnQWxhc2thJywgJ0hhd2FpaScpKSAjIGZyb20gYWxiZXJzdWENCmxlYWZsZXRfc2YgJTw+JSBnZW9fam9pbihjcm9zc1NlY3Rpb25hbERhdGEsIGJ5X3NwPSAnZmlwcycsIGJ5X2RmPSAna2V5X2xvY2FsJykgJT4lIA0KICBsZWZ0X2pvaW4oY2x1c3RlckNvdW50aWVzWywgYygna2V5X2xvY2FsJywgJ2NsdXN0ZXJfZ3JvdXAnKV0sIGJ5ID0gYygnZmlwcycgPSAna2V5X2xvY2FsJykpDQoNCiMjIyMgU2V0dGluZyB0aGUgQ29sb3IgU2NoZW1lDQpsZWFmbGV0UGFsIDwtICBjb2xvckZhY3RvcignU2V0MicsIGRvbWFpbiA9IGxlYWZsZXRfc2YkY2x1c3Rlcl9ncm91cCwgbmEuY29sb3IgPSAid2hpdGUiKQ0KDQojIyMjIFRoZSB2aXN1YWwNCmxlYWZsZXQoaGVpZ2h0PTUwMCkgJT4lICMgaW5pdGlhbGl6aW5nIHRoZSBsZWFmbGV0IG1hcA0KICBzZXRWaWV3KGxuZyA9IC05NiwgbGF0ID0gMzcuOCwgem9vbSA9IDMuOCkgJT4lICMgc2V0dGluZyB0aGUgdmlldyBvbiBDb250aW5lbnRhbCBVUw0KICBhZGRUaWxlcygpICU+JSAjIGFkZGluZyB0aGUgZGVmYXVsdCB0aWxlcw0KICBhZGRQb2x5Z29ucyhkYXRhID0gbGVhZmxldF9zZiwgc3Ryb2tlID0gRkFMU0UsIGZpbGxDb2xvciA9IH5sZWFmbGV0UGFsKGxlYWZsZXRfc2YkY2x1c3Rlcl9ncm91cCksICMgYWRkaW5nIHRoZSBkYXRhDQogICAgICAgICAgICAgIHdlaWdodCA9IDIsIG9wYWNpdHkgPSAxLCBjb2xvciA9ICJ3aGl0ZSIsIGRhc2hBcnJheSA9ICIzIiwgZmlsbE9wYWNpdHkgPSAwLjcsICMgYWRkaW5nIGNvbG9yIHNwZWNzDQogICAgICAgICAgICAgIHBvcHVwID0gcGFzdGUoIkNvdW50eToiLCBsZWFmbGV0X3NmJG5hbWUsICc8YnI+JywgDQogICAgICAgICAgICAgICAgICAgICAgICAgICAgIkNsdXN0ZXIgIzoiLCBsZWFmbGV0X3NmJGNsdXN0ZXJfZ3JvdXAsICc8YnI+JywNCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAiUG9wdWxhdGlvbiBEZW5zaXR5OiIsIHJvdW5kKGxlYWZsZXRfc2YkZV9wb3BkZW5zaXR5LCAxKSwgJzxicj4nKSkgJT4lICNwb3AtdXAgTWVudQ0KICBhZGRMZWdlbmQocG9zaXRpb24gPSAiYm90dG9tbGVmdCIsIHBhbCA9IGxlYWZsZXRQYWwsIHZhbHVlcyA9ICBsZWFmbGV0X3NmJGNsdXN0ZXJfZ3JvdXAsIA0KICAgICAgICAgICAgdGl0bGUgPSAiQ2x1c3RlciAjIiwgb3BhY2l0eSA9IDEpICMgbGVnZW5kIGZvcm1hdHRpbmcNCmBgYA0KDQoNCiMgRXhwbGFuYXRvcnkgTW9kZWxpbmcgb2YgQ2x1c3RlciBBc3NpZ25tZW50cyANCg0KSW4gdGhlIHByZXZpb3VzIHNlY3Rpb24sIHdlIHNob3dlZCB0aGF0IGJ5IHVzaW5nIHNvbGVseSBhIHNjYWxlZCBhbmQgc21vb3RoZWQgdGltZSBzZXJpZXMgb2YgZGFpbHkgZGVhdGhzIHBlciBjb3VudHksIHRoZSBjb3VudGllcyBhcmUgZ3JvdXBlZCBpbnRvIGByIHVuaXF1ZShjdHlfc2YkY2x1c3Rlcl9ncm91cCkgJT4lIG5hLm9taXQoKSAlPiUgbGVuZ3RoKClgIGNhdGVnb3JpZXMgKHdob3NlIHRpbWUtc2VyaWVzIGhhdmUgZGlzdGluY3Qgc2hhcGVzIGJhc2VkIG9uIHRoZSBFdWNsaWRlYW4gZGlzdGFuY2UgbWVhc3VyZSkuIEluIHRoaXMgc2VjdGlvbiwgd2UgYXR0ZW1wdCB0byBtb2RlbCB0aGUgZmFjdG9ycyB0aGF0IGFyZSBhc3NvY2lhdGVkIHdpdGggdGhlIGNsdXN0ZXIgYXNzaWdubWVudC4NCg0KIyMgRGVzY3JpcHRpdmUgU3RhdGlzdGljcw0KDQojIyMgU2tpbW1pbmcgdGhlIERhdGENCg0KYGBge3Igc2tpbSwgb3V0LndpZHRoPScxMDAlJywgcmVzdWx0cz0nYXNpcyd9DQojIENvbWJpbmluZyBtdWx0aUNsYXNzIHJlc3BvbnNlIHdpdGggcG90ZW50aWFsIHByZWRpY3RvcnMNCm11bHRpQ2xhc3NERiA8LSBzZWxlY3QoY2x1c3RlckNvdW50aWVzLCBrZXlfbG9jYWwsIGNsdXN0ZXJfZ3JvdXApICU+JSANCiAgbGVmdF9qb2luKGNyb3NzU2VjdGlvbmFsRGF0YSwgYnkgPSAna2V5X2xvY2FsJykgICU+JSANCiAgc2VsZWN0KC1jKGVfdG90cG9wLCBhcmVhX3NxbWksIHN0YXRlLCBjb3VudHkpKQ0KDQpzYXZlUkRTKG11bHRpQ2xhc3NERiwgJy4uL0RhdGEvT3V0cHV0L211bHRpQ2xhc3NERi5yZHMnKSAjIHNhdmluZyB0aGUgZGF0YQ0KDQpza2ltKG11bHRpQ2xhc3NERikgIyBwcmludGluZyBhIG5pY2Ugc3VtbWFyeSB0YWJsZSBvZiB0aGUgZGF0YQ0KYGBgDQoNCiMjIyBDb3JyZWxhdGlvbiBQbG90IEFtb25nIENvbnRpbnVvdXMgUHJlZGljdG9ycw0KDQpgYGB7ciBjb3JyUGxvdH0NCnRpZmYoZmlsZW5hbWUgPSAnLi4vRmlndXJlcy9jb3JyUGxvdC50aWZmJywNCiAgICB3aWR0aCA9IDEzNjYsIGhlaWdodCA9NzY4LCBwb2ludHNpemUgPSAxNikNCm5hLm9taXQobXVsdGlDbGFzc0RGKSAlPiUgDQogIHBsb3RfY29ycmVsYXRpb24oZ2d0aGVtZSA9IHRoZW1lX2J3KCksIHR5cGUgPSAnYycpICAjIGNvbXB1dGUgY29yciBhbW9uZyBvbmx5IGNvbnRpbnVvdXMgdmFycw0KaW52aXNpYmxlKCBkZXYub2ZmKCkgKSAjIHRvIHN1cHByZXNzIHRoZSB1bndhbnRlZCBvdXRwdXQgZnJvbSBkZXYub2ZmDQoNCiMgUHJpbnRpbmcgYSBwbmcgdmVyc2lvbiBvZiB0aGUgcGxvdCBpbiBNYXJrZG93biAobG93ZXIgcXVhbGl0eSBpbWFnZSBmb3IgcXVpY2tlciBjb21waWxhdGlvbiBvZiBIVE1MKQ0KcmVhZFRJRkYoIi4uL0ZpZ3VyZXMvY29yclBsb3QudGlmZiIpICU+JSBncmlkLnJhc3RlcigpDQpgYGANCg0KIyMjIEJveHBsb3QgQnkgQ2x1c3Rlcg0KDQpgYGB7ciBib3hQbG90fQ0KIyBTYXZpbmcgY3VtdWxhdGl2ZSBkZWF0aHMgZmlndXJlIHRvIGEgdGlmZiBmaWxlDQp0aWZmKGZpbGVuYW1lID0gJy4uL0ZpZ3VyZXMvYm94UGxvdC50aWZmJywNCiAgICB3aWR0aCA9IDEzNjYsIGhlaWdodCA9NzY4LCBwb2ludHNpemUgPSAxNikNCm11bHRpQ2xhc3NERiAlPiUgDQogIHBsb3RfYm94cGxvdChieSA9ICdjbHVzdGVyX2dyb3VwJywgbmNvbCA9IDJMLCANCiAgICAgICAgICAgICAgIGdndGhlbWUgPSB0aGVtZV9idygpLA0KICAgICAgICAgICAgICAgZ2VvbV9ib3hwbG90X2FyZ3MgPSBsaXN0KCdvdXRsaWVyLnNoYXBlJyA9IDEpKQ0KaW52aXNpYmxlKCBkZXYub2ZmKCkgKSAjIHRvIHN1cHByZXNzIHRoZSB1bndhbnRlZCBvdXRwdXQgZnJvbSBkZXYub2ZmDQoNCiMgUHJpbnRpbmcgYSBwbmcgdmVyc2lvbiBvZiB0aGUgcGxvdCBpbiBNYXJrZG93biAobG93ZXIgcXVhbGl0eSBpbWFnZSBmb3IgcXVpY2tlciBjb21waWxhdGlvbiBvZiBIVE1MKQ0KcmVhZFRJRkYoIi4uL0ZpZ3VyZXMvYm94UGxvdC50aWZmIikgJT4lIGdyaWQucmFzdGVyKCkNCmBgYA0KDQoNCg0KIyMgRXhwbGFuYXRvcnkgTW9kZWxpbmcgVXNpbmcgTXVsdGlub21pYWwgUmVncmVzc2lvbg0KDQojIyMgTW9kZWwgQnVpbGRpbmcNCg0KYGBge3IgbXVsdGlub219DQpkZiA8LSBtdWx0aUNsYXNzREYgJT4lIHNlbGVjdCgtYyhrZXlfbG9jYWwsIGxvY2F0aW9uKSkgJT4lIA0KICBtdXRhdGUoY2x1c3Rlcl9ncm91cCA9IGFzLmZhY3RvcihjbHVzdGVyX2dyb3VwKSkNCg0KZGYkZV9wb3BkZW5zaXR5IDwtIGxvZyhkZiRlX3BvcGRlbnNpdHkpICMgc2luY2UgaXQgaXMgaGlnaGx5IHNrZXdlZCBhbmQgd2UgYXJlIHVzaW5nIGEgbGluZWFyIG1vZGVsIA0KDQojIHNldHRpbmcgdGhlIHJlZmVyZW5jZSBsZXZlbCB0byBjYXRlZ29yeSB3aXRoIG1heCBmcmVxdWVuY3kNCmRmJGNsdXN0UmVMZXZlbGVkIDwtICByZWxldmVsKGRmJGNsdXN0ZXJfZ3JvdXAsIHJlZiA9IG1heENhdChkZiRjbHVzdGVyX2dyb3VwKSApDQpkZiAgPC0gIGRmICU+JSBzZWxlY3QoLWMoY2x1c3Rlcl9ncm91cCkpICMgcmVtb3ZlZCBzaW5jZSB3ZSBjcmVhdGVkIGEgcmVMZXZlbGVkIHZlcnNpb24gYW5kIHN0b3JlZCBpdCBpbiBhIGRpZmYgdmENCg0KZmluYWxNb2RlbCA8LSAgcXVpZXQobXVsdGlub20oY2x1c3RSZUxldmVsZWQgfiAuLCBkYXRhID0gZGYpKSAjIGNyZWF0ZSBtb2RlbA0KYGBgDQoNCiMjIyBSZXN1bHRpbmcgTW9kZWwNCmBgYHtyIG11bHRpTW9kZWwyLCByZXN1bHRzPSdhc2lzJywgZGVwZW5kc29uID0gIm11bHRpbm9tIn0NCiMgU2F2aW5nIHRoZSByZXN1bHRzIGFzIGEgbGF0ZXggdGFibGUsIGJ1dCBub3QgcHJpbnRpbmcgaXQgb3V0IGluIHRoZSBNYXJrZG93biBkb2N1bWVudA0KaW52aXNpYmxlKHN0YXJnYXplcihmaW5hbE1vZGVsLCB0eXBlID0gJ2xhdGV4JywgcC5hdXRvID0gRkFMU0UsIG91dD0iLi4vRGF0YS9PdXRwdXQvbXVsdGkudGV4IiwgDQogICAgICAgICAgICAgICAgICAgIHNpbmdsZS5yb3cgPSBUUlVFLCBoZWFkZXIgPSBGQUxTRSkpDQoNCiMgdGFidWxhdGluZyBtb2RlbCByZXN1bHRzIGFzIGFuIEhUTUwsIHdoaWNoIHdlIHByaW50IGJlbG93DQpzdGFyZ2F6ZXIoZmluYWxNb2RlbCwgdHlwZSA9ICdodG1sJywgcC5hdXRvID0gRkFMU0UsIG91dD0iLi4vRGF0YS9PdXRwdXQvbXVsdGkuaHRtbCIsIHNpbmdsZS5yb3cgPSBGQUxTRSkNCmBgYA0KDQojIyMgTW9kZWwncyBQZXJmb3JtYW5jZQ0KYGBge3IgcHJlZGljdGl2ZVBlcmZvcm1hbmNlLCBkZXBlbmRzb24gPSAibXVsdGlub20ifQ0KIyBleGFtaW5pbmcgaG93IHdlbGwgdGhlIG1vZGVsIHBlcmZvcm1lZCBvbiBvdXIgZW50aXJlIGRhdGFzZXQNCiMgUmVjYWxsIHRoYXQgd2UgYXJlIGZpdHRpbmcgYW4gZXhwbGFuYXRvcnkgbW9kZWwsIGFuZCBub3QgYSBwcmVkaWN0aXZlIG1vZGVsDQpwcmVkaWN0ZWRDbGFzcyA8LSBwcmVkaWN0KGZpbmFsTW9kZWwsIGRmKQ0KDQojIENvbXB1dGluZyB0aGUgQ29uZnVzaW9uIE1ldHJpY3MgYW5kIEJ5IENsYXNzIE1ldHJpY3MNCmNvbmZNYXRyaXggPSBjb25mdXNpb25NYXRyaXgocHJlZGljdGVkQ2xhc3MsIGRmJGNsdXN0UmVMZXZlbGVkKQ0Kc2F2ZVJEUyhjb25mTWF0cml4LCAnLi4vRGF0YS9PdXRwdXQvY29uZk1hdHJpeC5yZHMnKSAjIHNhdmluZyB0aGUgZGF0YQ0KDQojIFByaW50aW5nIHRoZSBSZXN1bHRpbmcgVGFibGVzIE5pY2VseQ0KcGFuZGVyKGNvbmZNYXRyaXgkdGFibGUpDQpwYW5kZXIoY29uZk1hdHJpeCRieUNsYXNzKQ0KcGFuZGVyKGNvbmZNYXRyaXgkb3ZlcmFsbCkNCmBgYA0KDQojIyMgVmlzdWFsaXppbmcgdGhlIE1vZGVsJ3MgUHJlZGljdGlvbnMNCg0KYGBge3IgY2x1c3Rlck1hdGNoLCBkZXBlbmRzb24gPSAibXVsdGlub20ifQ0KcHJlZGljdGVkUHJvYnMgPC0gZml0dGVkKGZpbmFsTW9kZWwpICMgY29tcHV0aW5nIHByZWRpY3RlZCBwcm9iYWJpbGl0aWVzIGZvciBlYWNoIG9mIHRoZSBjbHVzdGVyIG91dGNvbWUgbGV2ZWxzDQptYXBSZXN1bHRzIDwtIGNiaW5kKG5hLm9taXQobXVsdGlDbGFzc0RGKSwgcHJlZGljdGVkUHJvYnMpICMgY29sIGJpbmRpbmcgcHJlZFByb2JzIGZvciBFYWNoIENsdXN0ZXIgd2l0aCBtdWx0aUNsYXNzREYNCg0KIyBGaW5kaW5nIGluZGljZXMgdG8gc3Vic2V0IHRoZSBkYXRhDQpudW1iZXJPZkNsdXN0ZXJzIDwtIHVuaXF1ZShtYXBSZXN1bHRzJGNsdXN0ZXJfZ3JvdXApICU+JSBhcy5jaGFyYWN0ZXIoKSAlPiUgbGVuZ3RoKCkgDQpzdGFydENvbCA8LSBuY29sKG1hcFJlc3VsdHMpIC0gbnVtYmVyT2ZDbHVzdGVycyArIDENCmVuZENvbCA8LSBuY29sKG1hcFJlc3VsdHMpDQoNCiMgRmluZGluZyB3aGV0aGVyIHRoZSBwcmVkaWN0ZWQgYW5kIGFjdHVhbCBjbHVzdGVycyBtYXRjaGVkIGZvciBlYWNoIGNvdW50eQ0KbWFwUmVzdWx0cyRMYXJnZXN0UHJvYkNsdXN0ZXIgPC0gY29sbmFtZXMobWFwUmVzdWx0c1ssIHN0YXJ0Q29sOmVuZENvbF0pW2FwcGx5KG1hcFJlc3VsdHNbLCBzdGFydENvbDplbmRDb2xdLCAxLCB3aGljaC5tYXgpXSANCm1hcFJlc3VsdHMkbWF0Y2ggPC0gaWZlbHNlKG1hcFJlc3VsdHMkY2x1c3Rlcl9ncm91cCA9PSBtYXBSZXN1bHRzJExhcmdlc3RQcm9iQ2x1c3RlciwgJ1llcycsICdObycpICU+JSBhcy5mYWN0b3IoKQ0KDQojIFJldHJpZXZpbmcgdGhlIFUuUy4gY291bnR5IGNvbXBvc2l0ZSBtYXAgYXMgYSBzaW1wbGVmZWF0dXJlIChzaW5jZSBpdCBoYXMgYmVlbiBvdmVyd3JpdHRlbikNCmN0eV9zZiA9IGdldF91cmJuX21hcChtYXAgPSAiY291bnRpZXMiLCBzZiA9IFRSVUUpICU+JSANCiAgZmlsdGVyKCFzdGF0ZV9uYW1lICVpbiUgYygnQWxhc2thJywgJ0hhd2FpaScpKQ0KDQpjdHlfc2YgJTw+JSBnZW9fam9pbihtYXBSZXN1bHRzLCBieV9zcD0gJ2NvdW50eV9maXBzJywgYnlfZGY9ICdrZXlfbG9jYWwnKQ0KDQojIENyZWF0aW5nIGEgc3RhdGljIHZpc3VhbCBmb3IgdGhlIHBhcGVyDQpwZGYoZmlsZSA9ICcuLi9GaWd1cmVzL2NsdXN0ZXJNYXRjaE1hcC5wZGYnLCB3aWR0aCA9IDYuNSwgaGVpZ2h0ID0gNi41KQ0KdG1fc2hhcGUoY3R5X3NmKSArIA0KICB0bV9wb2x5Z29ucygnbWF0Y2gnLCB0aXRsZSA9ICdDbHVzdGVyIE1hdGNoJywgc3R5bGUgPSAnY29udCcsIHBhbGV0dGUgPSAiZGl2IikgKw0KICB0bV9sYXlvdXQoYWVzLnBhbGV0dGUgPSBsaXN0KGRpdiA9IGxpc3QoIlllcyIgPSAiI0ZGRkZGRiIsICJNaXNzaW5nIiA9ICIjQkRCREJEIiwgIk5vIiA9ICIjMDAwMDAwIikpKSArDQogIHRtX2xheW91dChvdXRlci5tYXJnaW5zID0gYygwLDAsMCwwKSwgZnJhbWUgPSBGLCBsZWdlbmQudGV4dC5zaXplID0gMSkNCmludmlzaWJsZSggZGV2Lm9mZigpICkgIyB0byBzdXBwcmVzcyB0aGUgdW53YW50ZWQgb3V0cHV0IGZyb20gZGV2Lm9mZg0KDQoNCiMgQ3JlYXRpbmcgYW4gaW50ZXJhY3RpdmUgdmlzdWFsIFVzaW5nIHRoZSBMZWFmbGV0IFBhY2thZ2UNCiMjIyMgQ3JlYXRpbmcgYSBsb25nbGF0IHByb2plY3Rpb24gKHJlcXVpcmVkIGJ5IGxlYWZsZXQpDQpsZWFmbGV0X3NmIDwtIGNvdW50aWVzX3NmKCJsb25nbGF0IikgJT4lIGZpbHRlcighc3RhdGUgJWluJSBjKCdBbGFza2EnLCAnSGF3YWlpJykpICMgZnJvbSBhbGJlcnN1YQ0KbGVhZmxldF9zZiAlPD4lIGdlb19qb2luKG1hcFJlc3VsdHMsIGJ5X3NwPSAnZmlwcycsIGJ5X2RmPSAna2V5X2xvY2FsJykNCg0KIyMjIyBTZXR0aW5nIHRoZSBDb2xvciBTY2hlbWUNCmxlYWZsZXRQYWwgPC0gIGNvbG9yRmFjdG9yKHBhbGV0dGUgPSBjKCIjQ0FCMkQ2IiwgIiM2QTNEOUEiKSwgbGV2ZWxzID0gYygnWWVzJywgJ05vJyksIG5hLmNvbG9yID0gIndoaXRlIikNCg0KIyMjIyBUaGUgdmlzdWFsDQpsZWFmbGV0KGhlaWdodD01MDApICU+JSAjIGluaXRpYWxpemluZyB0aGUgbGVhZmxldCBtYXANCiAgc2V0VmlldyhsbmcgPSAtOTYsIGxhdCA9IDM3LjgsIHpvb20gPSAzLjgpICU+JSAjIHNldHRpbmcgdGhlIHZpZXcgb24gQ29udGluZW50YWwgVVMNCiAgYWRkVGlsZXMoKSAlPiUgIyBhZGRpbmcgdGhlIGRlZmF1bHQgdGlsZXMNCiAgYWRkUG9seWdvbnMoZGF0YSA9IGxlYWZsZXRfc2YsIHN0cm9rZSA9IEZBTFNFLCBmaWxsQ29sb3IgPSB+bGVhZmxldFBhbChsZWFmbGV0X3NmJG1hdGNoKSwgIyBhZGRpbmcgdGhlIGRhdGENCiAgICAgICAgICAgICAgd2VpZ2h0ID0gMiwgb3BhY2l0eSA9IDEsIGNvbG9yID0gIndoaXRlIiwgZGFzaEFycmF5ID0gIjMiLCBmaWxsT3BhY2l0eSA9IDAuNywgIyBhZGRpbmcgY29sb3Igc3BlY3MNCiAgICAgICAgICAgICAgcG9wdXAgPSBwYXN0ZSgiQ291bnR5OiIsIGxlYWZsZXRfc2YkbmFtZSwgJzxicj4nLCANCiAgICAgICAgICAgICAgICAgICAgICAgICAgICAiQ2x1c3RlciAjOiIsIGxlYWZsZXRfc2YkY2x1c3Rlcl9ncm91cCwgJzxicj4nLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICJDbHVzdGVyIFByZWRpY3RlZDoiLCBsZWFmbGV0X3NmJExhcmdlc3RQcm9iQ2x1c3RlciwgJzxicj4nLA0KICAgICAgICAgICAgICAgICAgICAgICAgICAgICJDbHVzdGVyIE1hdGNoOiIsIGxlYWZsZXRfc2YkbWF0Y2gsICc8YnI+JykpICU+JSAjcG9wLXVwIE1lbnUNCiAgYWRkTGVnZW5kKHBvc2l0aW9uID0gImJvdHRvbWxlZnQiLCBwYWwgPSBsZWFmbGV0UGFsLCB2YWx1ZXMgPSAgbGVhZmxldF9zZiRtYXRjaCwgDQogICAgICAgICAgICB0aXRsZSA9ICJDbHVzdGVyIE1hdGNoIiwgb3BhY2l0eSA9IDEpICMgbGVnZW5kIGZvcm1hdHRpbmcNCmBgYA0KDQojIyMgQSBOb3RlIG9uIE91ciBQcmVkaWN0aXZlIFBlcmZvcm1hbmNlDQpJbiB0aGUgY29kZSBjaHVuayBiZWxvdywgd2UgY2FwaXRhbGl6ZSBvbiB0aGUgW011TUluIHBhY2thZ2VdKGh0dHBzOi8vY3Jhbi5yLXByb2plY3Qub3JnL3dlYi9wYWNrYWdlcy9NdU1Jbi9NdU1Jbi5wZGYpIHRvIGdlbmVyYXRlIGEgbW9kZWwgc2VsZWN0aW9uIHRhYmxlIG9mIG1vZGVscyB3aXRoIGNvbWJpbmF0aW9ucyAoc3Vic2V0cykgb2YgZml4ZWQgZWZmZWN0IHRlcm1zIGluIHRoZSBnbG9iYWwgbW9kZWwuIE1vcmUgc3BlY2lmaWNhbGx5LCB0aGUgY29kZSBiZWxvdyBpZGVudGlmaWVzIHRoZSBjb21iaW5hdGlvbiBvZiBwcmVkaWN0b3JzIChhbmQgaGVuY2UgdGhlIG51bWJlciBvZiBtb2RlbCBwYXJhbWV0ZXJzKSB0aGF0IHdvdWxkIG1pbmltaXplIHRoZSAqQUlDKi4NCmBgYHtyIG11bHRpTW9kZWxJbmZlcmVuY2UsIGRlcGVuZHNvbiA9ICJtdWx0aW5vbSJ9DQpkZk5vTkFzIDwtIG5hLm9taXQoZGYpICMgc2luY2UgdGhlIE11TUluIHBhY2thZ2UgZXhwZWN0cyBubyBOQXMgaW4gdGhlIGRhdGEgZnJhbWUNCmZtIDwtICBxdWlldChtdWx0aW5vbShjbHVzdFJlTGV2ZWxlZCB+IC4sIGRhdGEgPSBkZk5vTkFzLCBuYS5hY3Rpb24gPSBuYS5mYWlsKSkgIyBjcmVhdGUgbW9kZWwNCm1zIDwtIHF1aWV0KGRyZWRnZShmbSwgcmFuayA9ICdBSUMnKSkgIyBnZW5lcmF0aW5nIHRoZSBtb2RlbCBzZWxlY3Rpb24gdGFibGUNCg0KIyBQcmludGluZyB0aGUgbW9kZWxzIHdpdGggdGhlIGxvd2VzdCBBSUMgZmlyc3QNCmRhdGF0YWJsZShtcyAlPiUgYXMuZGF0YS5mcmFtZSgpICU+JSBzZWxlY3QoLWMoZGVsdGEsIHdlaWdodCkpLA0KICAgICAgICAgIGV4dGVuc2lvbnMgPSBjKCdGaXhlZENvbHVtbnMnLCAnQnV0dG9ucycpLCBvcHRpb25zID0gbGlzdCgNCiAgICAgICAgICAgIGRvbSA9ICdCZnJ0aXAnLA0KICAgICAgICAgICAgc2Nyb2xsWCA9IFRSVUUsDQogICAgICAgICAgICBidXR0b25zID0gYygnY29weScsICdjc3YnLCAnZXhjZWwnLCAncGRmJywgJ3ByaW50JyksDQogICAgICAgICAgICBmaXhlZENvbHVtbnMgPSBsaXN0KGxlZnRDb2x1bW5zID0gMCwgcmlnaHRDb2x1bW5zID0gMykpLA0KICAgICAgICAgIHJvd25hbWVzID0gRkFMU0UpICU+JSANCiAgZm9ybWF0Um91bmQoY29sdW1ucz0gYygnbG9nTGlrJywgJ0FJQycpLCBkaWdpdHM9MCkNCmBgYA0KV2l0aCB0aGUgdXNlIG9mIGFuIGFsdGVybmF0aXZlIG1vZGVsIHNlbGVjdGlvbiBjcml0ZXJpb24gYW5kIHRoaXMgZXZhbHVhdGlvbiwgd2UgY29uZmlybSB0aGF0IHRoZSBvcHRpbWFsIG11bHRpbm9taWFsIHJlZ3Jlc3Npb24gbW9kZWw6ICANCg0KICAtIGluY2x1ZGVzIHRoZSB2YXJpYWJsZXMgZGVub3RlZCB3aXRoIGEgJysnIHNpZ24gKGluIHRoZSBmaXJzdCByb3cgb2YgdGhlIHByaW50ZWQgdGFibGUpOyAgDQogIC0gY29udGFpbnMgYHIgcm91bmQobXNbMSxdICU+JSBhcy5kYXRhLmZyYW1lKCkgJT4lIHNlbGVjdChkZiksIDApYCBwYXJhbWV0ZXJzIGluIHRoZSBtb2RlbDsgIA0KICAtIHJlc3VsdHMgaW4gYSBsb2dMaWsgdmFsdWVzIG9mIGByIHJvdW5kKG1zWzEsXSAlPiUgYXMuZGF0YS5mcmFtZSgpICU+JSBzZWxlY3QobG9nTGlrKSwgMClgOyBhbmQgIA0KICAtIHJlc3VsdHMgaW4gYSBtaW5pbXVtIEFJQyBvZiBgciByb3VuZChtc1sxLF0gJT4lIGFzLmRhdGEuZnJhbWUoKSAlPiUgc2VsZWN0KEFJQyksIDApYC4gIA0KDQoNCg0KIyBDb25jbHVkaW5nIFJlbWFya3MNCkluIHRoaXMgUiBNYXJrZG93biBkb2N1bWVudCwgd2UgaGF2ZSBzaG93biB0aGF0IG91ciBwcm9wb3NlZCB0d28gc3RhZ2UgZnJhbWV3b3JrIGZvciBtb2RlbGluZyB0aGUgc21vb3RoZWQgYW5kIHNjYWxlZCB0aW1lIHNlcmllcyBvZiBuZXcgZGFpbHkgY2FzZXMgY2FuIHByb3ZpZGUgaW5zaWdodHMgaW50byB0aGUgc2hhcGUgb2YgdGhlIG91dGJyZWFrJ3MgdGltZS1zZXJpZXMgYW5kIHNvbWUgb2YgaXRzIGFzc29jaWF0ZWQgZmFjdG9ycy4gU3BlY2lmaWNhbGx5LCB3ZSBoYXZlIHNob3duIHRoYXQ6ICANCg0KLSBPbiBhIGNvdW50eS1sZXZlbCwgdGhlIHRpbWUgc2VyaWVzIG9mIENPVklELTE5IG5ldyBkYWlseSBjYXNlcyBjYW4gYmUgY2x1c3RlcmVkIGludG8gYHIgdW5pcXVlKG1hcFJlc3VsdHMkY2x1c3Rlcl9ncm91cCkgJT4lIGxlbmd0aCgpYCBjbHVzdGVycy4gIA0KDQotIFVzaW5nIGEgbXVsdGlub21pYWwgcmVncmVzc2lvbiBtb2RlbCwgd2UgaGF2ZSBzaG93bi9xdWFudGlmaWVkIHRoZSBpbXBhY3Qgb2YgdGhlIGZvbGxvd2luZyBmYWN0b3JzOiBgciBzZWxlY3QobXVsdGlDbGFzc0RGLCAtYyhrZXlfbG9jYWwsIGNsdXN0ZXJfZ3JvdXAsIGxvY2F0aW9uKSkgJT4lIGNvbG5hbWVzKCkgJT4lIHBhbmRlcigpYCBvbiB0aGUgb2RkcyBvZiBiZWluZyBpbiBhIHNwZWNpZmljIGNsdXN0ZXIgd2hlbiBjb21wYXJlZCB0byB0aGUgYmFzZWxpbmUuDQoNCg0KLS0tDQoNCiMgUmVmZXJlbmNlcw0KPGRpdiBpZD0icmVmcyI+PC9kaXY+DQoNCi0tLQ0KDQojIEFwcGVuZGl4DQpJbiB0aGlzIGFwcGVuZGl4LCB3ZSBwcmludCBhbGwgdGhlIFIgcGFja2FnZXMgdXNlZCBpbiBvdXIgYW5hbHlzaXMgYW5kIHRoZWlyIHZlcnNpb25zIHRvIGFzc2lzdCB3aXRoIHJlcHJvZHVjaW5nIG91ciByZXN1bHRzL2FuYWx5c2lzLg0KDQpgYGB7ciBzZXNzaW9uSW5mb30NCnBhbmRlcihzZXNzaW9uSW5mbygpLCBjb21wYWN0ID0gVFJVRSkgIyBwcmludGluZyB0aGUgc2Vzc2lvbiBpbmZvDQpgYGA=